Effective C++(5) 了解C++默默地编写并调用哪些函数

预热:

一个空的类,当编译器处理过之后,就包含:
  • 一个copy构造函数
  • 一个重载赋值操作符
  • 一个析构函数
  • 一个默认构造函数
Demo:

class Empty()  {   };       // 声明一个空的类

class Empty()     // 经过编译器处理后
{
public:
    Empty() { ... }
    Empty(const Empty& rhs) { ... }
    ~Empty() { ... }

    Empty& operator=(const Empty& rhs) { ... }
};


需要注意的一点是,只有当这些函数被需要时,它们才会被编译器创建出来。

现在我们知道了,编译器会为我们创建这些函数。
下面举个例子引入今天的重点问题
如果编译器拒绝为我们做这些事情了,就麻烦了。那么,编译器什么情况下会拒绝为我们做这些事情呢?

Demo:

class Empty()  {   };       // 声明一个空的类

class Empty()     // 经过编译器处理后
{
public:
    Empty() { ... }
    Empty(const Empty& rhs) { ... }
    ~Empty() { ... }

    Empty& operator=(const Empty& rhs) { ... }
};
 


类NameObject并没有给我们重载赋值操作符,哪门这个编译器会给我提供这个操作吗,如果提供了这个操作,那么Demo中的p最后的成员的值是什么呢?让我们慢慢来分析一下:
1 p.nameValue
C++规定:不可以让reference改指向不同的对象。
因此,C++对这种情况是拒绝编译这一行的赋值动作。即如果你打算在一个有reference成员的class内支持重载赋值操作,那么你必须自己定义重载赋值操作符。
2 p.objectValue
对于const成员变量,更改const变量是不合法的,所以编译器同样拒绝为你创建这个赋值操作。
3 还有一个情况会导致编译器“罢工”
如果某个base classes将重载赋值操作符声明为private。
原因:子类中的重载赋值操作符会默认去调用基类中的赋值操作符。

小结:
编译器可以暗自为你创建默认构造函数、拷贝构造函数、重载赋值操作符,以及析构函数。
但是需要注意那些编译器拒绝你的情况。

参考:
《Effective C++ 3rd》

 

posted @ 2013-12-23 11:45  suzhou  阅读(225)  评论(0编辑  收藏  举报