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;
        }