effective c++(05)(06)之c++默默编写并调用的函数

1. 当只写一个空类的时候,编译器会为他声明一个copy构造函数,一个copy assignment函数和一个析构函数。如下:

如果写下:

class Empty{ };

编译器就会实现以下代码:

class Empty{
    Empty(){}  //默认构造函数
    Empty(const Empty& rhs ) {}  //复制构造函数
~Empty() {} //析构函数 Empty& operator=( const Empty& rhs ) {} // 复制赋值操作符
};

编译器产生的是non-virtual函数。默认产生以上四种构造函数。

如果要使这些编译器默认产生的函数失效则提供了以下两种方法:

一. 将这些函数声明为private。如下:

class Empty{
public:
private:
    Empty(const Empty& rhs ) {}
};

此时如果进行如下调用:

    Empty e1;
    Empty e2(e1);
    return 0;

编译器就不会允许。但是这种情况在member函数或者friend函数内进行的时候就可以做到。

二. 为了能够保证所有的成员以及类不能够拷贝对象,需要定义以下类:

class Uncopyable{
protected:
    Uncopyable() {}
    ~Uncopyable(){}
private:
    Uncopyable( const Uncopyable& );
    Uncopyable& operator=( const Uncopyable& );
};

然后使

class Empty:private Uncopyable{
    .......
};

这样,当任何人(甚至是member函数或者friend函数)尝试拷贝Empty对象时,编译器试着生成一个copy构造函数和一个copy assignment操作符,而这些函数却尝试着调用base class对应的兄弟,此时编译器便拒绝操作。

2.

class NamedObject{
public:
    NamedObject( const char* name, const T& value );
    NamedObject( const string& name, const T& value );

private:
    string nameValue;
    T objectValue;
};
NamedObject<int> no1( "Smallest Prim Number", 2 );
NamedObject<int> no2(no1);

编译器已经为NamedObject类定义了copy构造函数,所以  NamedObject<int> no2(no1)  就调用了该copy构造函数。

3.

template< typename T >
class NamedObject{
public:
    NamedObject( const string& name, const T& value );

private:
    string& nameValue;
    const T objectValue;
};
string newDog("persephone");
string oldDog("Satch");
NamedObject<int> p( newDog, 2 );
NamedObject<int> s( oldDog, 36 );
p = s;

c++不允许“让reference”指向不同的对象。如果打算在一个内含有“reference”成员的class内支持赋值操作,必须要定义一个copy assignment操作符。同样对于内含const的成员的类,更改const也是不合法的。

4. 如果某个base class将copy assignment声明为private,编译器也拒绝为其derived classes生成copy assignment操作符。因为编译器为derived classes生成的copy assignment操作符可以处理base class成分。

总结:编译器可以暗自为class创建default构造函数,copy构造函数,copy assignment操作符以及析构函数。

posted @ 2013-07-11 19:40  2011winseu  阅读(413)  评论(1编辑  收藏  举报