1.C++重载=(赋值运算符)
- 在定义的同时进行赋值叫做初始化(Initialization),定义完成以后再赋值(不管在定义的时候有没有赋值)就叫做赋值(Assignment)。初始化只能有一次,赋值可以有多次。当以拷贝的方式初始化一个对象时,会调用拷贝构造函数;当给一个对象赋值时,会调用重载过的赋值运算符。即使没有显式的重载赋值运算符,编译器也会以默认地方式重载它。默认重载的赋值运算符功能很简单,就是将原有对象的所有成员变量一一赋值给新对象,这和默认拷贝构造函数的功能类似。
- 对于简单的类,默认的赋值运算符一般就够用了,也没有必要再显式地重载它。但是当类持有其它资源时,例如动态分配的内存、打开的文件、指向其他数据的指针、网络连接等,默认的赋值运算符就不能处理了,我们必须显式地重载它,这样才能将原有对象的所有数据都赋值给新对象。实例如下:
#include "iostream" #include "cstdlib" #include "cstring" using namespace std; class Array{ public: Array(int len); Array(const Array &arr); // 拷贝构造函数 ~Array(); public: int operator[](int i) const {return m_p[i];} // 以常成员函数(只读)形式重载[]运算符 int & operator[](int i) {return m_p[i];} // 获取元素(写入) Array & operator=(const Array &arr); // 重载赋值运算符!!!operator=()的形参类型是const Array &,这样不但能够避免在传参时调用拷贝构造函数,还能够同时接收const类型和非const类型的实参 int length() const {return m_len;} void show(); private: int m_len; int *m_p; }; // 构造函数的定义 Array::Array(int len):m_len(len){ m_p = (int *)calloc(len, sizeof(int)); // 动态分配内存空间,使得指针m_p指向新分配的内存空间 } // 拷贝构造函数的定义 Array::Array(const Array &arr){ this->m_len = arr.m_len; this->m_p = (int *)calloc(this->m_len, sizeof(int)); memcpy( this->m_p, arr.m_p, m_len * sizeof(int)); } Array::~Array(){ free(m_p); } // 重载赋值运算符 Array & Array::operator=(const Array &arr){ // operator=()的返回值类型是Array &,这样不但能够避免在返回数据时调用拷贝构造函数,还能够达到连续赋值的目的 if(this != &arr){ // 判断是否是给同一个对象赋值 this->m_len = arr.m_len; free(this->m_p); this->m_p = (int*)calloc(this->m_len, sizeof(int)); memcpy(this->m_p, arr.m_p, m_len * sizeof(int)); } return *this; // 表示返回当前对象 } // 普通成员函数的定义 void Array::show(){ cout << "我是普通成员函数....\n"; } // 打印数组元素 void printArray(const Array &arr){ int len = arr.length(); for(int i=0; i<len; i++){ if(i == len-1){ cout<<arr[i]<<endl; }else{ cout<<arr[i]<<", "; } } } int main(){ Array arr1(10); for(int i=0; i<10; i++){ arr1[i] = i; } cout << "arr1:"; printArray(arr1); Array arr2(5); for(int i=0; i<5; i++){ arr2[i] = i; } cout << "arr2:"; printArray(arr2); cout << "------------------------------------\n"; arr2 = arr1; // 调用operator=() cout << "将arr1赋值给arr2:"; printArray(arr2); arr2[3] = 234; // 修改arr2的数据不会影响arr1,如果把operator=()注释掉,修改arr2的值就会影响arr1的值 arr2[7] = 920; cout << "修改arr2的值不会影响arr1:"; printArray(arr1); return 0; }