让编程改变世界
Change the world by program
MyClass obj1;
MyClass obj2;
obj2 = obj1;
前两行代码很简明,它们创建出了两个MyClass类的实例obj1和obj2。第三行代码把obj1的值赋值给了obj2,这里就可能会埋下祸根! 那么,怎样才能截获这个赋值操作并告诉它应该如何处理那些指针呢?MyClass &operator = (const MyClass &rhs);
上边的语句告诉我们这个方法所预期的输入参数应该是一个MyClass类型的、不可改变的引用。 因为这里使用的参数是一个引用,所以编译器在传递输入参数时就不会再为它创建另外一个副本(否则可能导致无限递归) 又因为这里只需要读取这个输入参数,而不用改变它的值,所以我们用const把那个引用声明为一个常量确保万无一失。 返回一个引用,该引用指向一个MyClass类的对象。如果看过我们待会实现的源码,可能会发觉这个没有必要。但是,这样确实是一个好习惯! 另外的好处是方便我们把一组赋值语句串联起来,如:a = b = c;MyClass obj1;
MyClass obj2 = obj1;
这与刚才那三行的区别很细微,刚才是先创建两个对象,然后再把obj1赋值给obj2。 现在是先创建一个实例obj1,然后再创建实例obj2的同时用obj1的值对它进行初始化。 虽然看起来好像一样,但编译器却生成完全不同的代码:编译器将在MyClass类里寻找一个副本构造器(copy constructor),如果找不到,它会自行创建一个。 即时我们对赋值操作符进行了重载,由编译器创建的副本构造器仍以”逐位复制”方式把obj1赋值给obj2。 换句话说,如果遇到上面这样的代码,即时已经在这个类里重载了赋值操作符,暗藏着隐患的”逐位复制”行为还是会发生。 想要躲开这个隐患,还需要亲自定义一个副本构造器,而不是让系统帮我们生成。MyClass( const MyClass &rhs);
这个构造器需要一个固定不变(const)的MyClass类型的引用作为输入参数,就像赋值操作符那样。因为他是一个构造器,所以不需要返回类型,还记得吗? 修改后:Example02.cpp(源代码下载) [buy] 获得所有教学视频、课件、源代码等资源打包 [/buy] [Downlink href='http://urlxf.qq.com/?rQfEbuR']视频下载[/Downlink] [Downlink href='http://kuai.xunlei.com/d/LLASZNHRFVKH']备胎下载[/Downlink]