【转】C++的赋值构造函数(赋值运算符重载)

当一个类的对象向该类的另一个对象赋值时,就会用到该类的赋值构造函数。

当没有重载赋值构造函数(赋值运算符)时,通过默认赋值构造函数来进行赋值操作

注意:这里a,b对象是已经存在的,是用a对象来赋值给b的。

赋值运算符的重载声明如下:

通常大家会对拷贝构造函数和赋值构造函数混淆,这里仔细比较两者的区别:

1)拷贝构造函数是一个对象初始化一块内存区域,这块内存就是新对象的内存区,而赋值构造函数时对于一个已经被初始化的对象来进行赋值操作。

2)实现不一样,拷贝构造函数首先是一个构造函数,它调用时候是通过参数的对象初始化产生一个新对象。赋值构造函数是把一个新的对象赋值给一个原有的对象。

举例:

结果:

说明:

1、参数

一般地,赋值运算符重载函数的参数是函数所在类的const类型的引用,加const是因为:

(1)我们不希望在这个函数中对用来进行赋值的”原版”做任何修改。

(2)加上const,对于const的和非const的实参,函数都能接受;如果不加,就只能接受非const的实参。

用引用是因为:

这样可以避免在函数调用时对实参的一次拷贝,提高了效率。

注意:上面的规定都不是强制的,可以不加const,也可以没有引用。

2、返回值

一般地,返回值是被赋值者的引用,即*this(如上面例1),原因是:

(1)这样在函数返回时避免一次拷贝,提高了效率。

(2)更重要的,这样可以实现连续赋值,即类似a=b=c这样。如果不是返回引用而是返回值类型,那么,执行a=b时,调用赋值运算符重载函数,在函数返回时,由于返回的是值类型,所以要对return后边的”东西”进行一次拷贝,得到一个未命名的副本(有些资料上称之为”匿名对象”),然后将这个副本返回,而这个副本是右值,所以,执行a=b后,得到的是一个右值,再执行=c就会出错。

注意:这也不是强制的,我们甚至可以将函数返回值声明为void,然后什么也不返回,只不过这样就不能够连续赋值了

3、赋值运算符重载函数不能被继承

因为相较于基类,派生类往往要添加一些自己的数据成员和成员函数,如果允许派生类继承积累的赋值运算符重载函数,那么,在派生类不提供自己的赋值运算符重载函数时,就只能调用基类的,但基类版本只能处理基类的数据成员,在这种情况下,派生类自己的数据成员怎么办?所以,C++规定,赋值运算符重载函数不能被继承。

4、赋值运算符重载函数要避免自赋值

对于赋值运算符重载函数,我们要避免自赋值(即自己给自己赋值)的发生,一般地,我们通过比较赋值者与被赋值者的地址是否相同来判断两者是否是同一对象(如例中的if(this != &str)一句)。避免自赋值的意义是:

(1)提高效率,显然,自己给自己赋值完全是毫无意义的无用功,特别地,对于基类数据成员间的赋值,还会调用基类的赋值运算符重载函数,开销是很大的。如果我们一旦判定是自赋值,就立即return *this,会避免对其它函数的调用。

(2)如果类的数据成员中含有指针,自赋值有时会导致灾难性的后果。对于指针间的赋值(注意这里指的是指针所指内容间的赋值,这里假设用_p给p赋值),先要将p所指向的空间delete掉(为什么要这么做呢?因为指针p所指的空间通常是new来的,如果在为p重新分配空间前没有将p原来的空间delete掉,会造成内存泄露),然后再为p重新分配空间,将_p所指的内容拷贝到p所指的空间。如果是自赋值,那么p和_p是同一指针,在赋值操作前对p的delete操作,将导致p所指的数据同时被销毁。那么重新赋值时,拿什么来赋?

所以,对于赋值运算符重载函数,一定要先检查是否是自赋值,如果是,直接return *this。

转载自:

Original: https://www.cnblogs.com/codingmengmeng/p/14116916.html
Author: 阿玛尼迪迪
Title: 【转】C++的赋值构造函数(赋值运算符重载)

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/535172/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • C++中的POD类型

    参考 定义 总结与理解 参考 https://en.cppreference.com/w/cpp/named_req/PODType 定义 知识的搬运工,以下内容抄的,虽然是硬性定…

    C++ 2023年5月29日
    064
  • 邻接表有向图(二)之 C++详解

    邻接表有向图是指通过邻接表表示的有向图。 上面的图G2包含了”A,B,C,D,E,F,G”共7个顶点,而且包含了” 上图右边的矩阵是G2在内存中…

    C++ 2023年5月29日
    065
  • C++ GUI Qt4编程(第二版) 源代码 下载

    C++ GUI Qt4编程(第二版) 源代码官方下载链接 Download the book examples for Windows (Zipped) Download the …

    C++ 2023年5月29日
    067
  • C++ STL std::copy 详解

    std::copy(start, end, std::back_inserter(container)); 这里,start和end是输入序列(假设有N个元素)的迭代器(itera…

    C++ 2023年5月29日
    060
  • c++builder调用VC的dll以及VC调用c++builder的dll

    解析__cdecl,__fastcall, __stdcall 的不同:在函数调用过程中,会使用堆栈,这三个表示不同的堆栈调用方式和释放方式。比如说__cdecl,它是标准的c方法…

    C++ 2023年5月29日
    065
  • [C++] const与重载

    下面的两个函数构成重载吗? cpp;gutter:true; void M(int a){} //(1) void M(const int a){} //(2)</p>…

    C++ 2023年5月29日
    083
  • C++11 并发指南五(std::condition_variable 详解)

    std::condition_variable 是条件变量,更多有关条件变量的定义参考维基百科。Linux 下使用 Pthread 库中的 pthread_cond_*() 函数提…

    C++ 2023年5月29日
    060
  • [转]C++重载()(强制类型转换运算符)

    在 C++ 中,类型的名字(包括类的名字)本身也是一种运算符,即类型强制转换运算符。 类型强制转换运算符是单目运算符,也可以被重载,但只能重载为成员函数,不能重载为全局函数。经过适…

    C++ 2023年5月29日
    059
  • c++自定义排序_lambda表达式

    class Solution { void quickSort(vector& strs, int l, int r) { if (l >= r) return; i…

    C++ 2023年5月29日
    046
  • js c++ 多值返回 返回多个值 c++ tuple

    使用C# 7.0推出的值元组和解构功能。 static (int, int) Calc(int a, int b) { return (a + b, a – b); } stati…

    C++ 2023年5月29日
    063
  • C++ 总结

    1、迭代器并不是都可以进行加减 迭代器实质上是一个指针,但是,并不是所有的容器的迭代器可以支持加减操作。 能进行算术运算的迭代器只有随机访问迭代器,要求容器元素存储在连续内存空间内…

    C++ 2023年5月29日
    064
  • android C/C++ source files 全局宏定义 .

    \system\core\include\arch\linux-arm AndroidConfig.h ======================================…

    C++ 2023年5月29日
    080
  • 生成1~n之间随机整数_c++

    rand() % (high – low + 1) + low; Original: https://www.cnblogs.com/douzujun/p/16457919.htm…

    C++ 2023年5月29日
    036
  • Cannot find a C++ compiler that supports both C++11 and the specified C++ flags.

    Linux 安装 cmake 时候出现的问题,解决方法: yum install gcc-c++ Original: https://www.cnblogs.com/hunttow…

    C++ 2023年5月29日
    054
  • C++11 并发指南六(atomic 类型详解一 atomic_flag 介绍)

    C++11 并发指南已经写了 5 章,前五章重点介绍了多线程编程方面的内容,但大部分内容只涉及多线程、互斥量、条件变量和异步编程相关的 API,C++11 程序员完全可以不必知道这…

    C++ 2023年5月29日
    050
  • 选择排序 C++实现

    实现思想: 1.寻找[i, n)区间里的最小值min ( i>= 0 ) 2.交换min和第i的数 ( i>= 0 ) #include #include using …

    C++ 2023年5月29日
    058
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球