【C++】explicit关键字

转载自:http://www.cnblogs.com/dwdxdy/archive/2012/07/17/2595479.html

C++提供关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换发生.

声明为explicit的构造函数不能在隐式转换中使用.

C++中,一个参数的构造函数(或者除了第一个参数外其余参数都有默认值的多参构造函数),承担了两个角色.

1.是个构造器,2.是个默认且隐含的类型转换操作符.

写下如AAA = XXX,这样的代码,且恰好XXX的类型正好是AAA单参数构造的参数类型,这时候编译器就自动调用这个构造器,创建一个AAA的对象.

使用explicit声明构造函数,则可防止隐式转换,避免上述情况的发生.具体例子如下:

class CTest1 {
public:
    CTest1(int n)
    {
        cout<<"Constructor of CTest1"<<endl;
    }
    CTest1(const CTest1&)
    {
        cout<<"Copy constructor of CTest1"<<endl;
    }
};
class CTest2 {
public:
    explicit CTest2(int n)
    {
        cout<<"Constructor of CTest2"<<endl;
    }
    explicit CTest2(const CTest2&)
    {
        cout<<"Copy constructor of CTest2"<<endl;
    }
};
int main()
{
    CTest1 a1(1);            //显示调用构造函数
    CTest1 b1 = 1;            //隐式调用构造函数
    CTest1 c1 = a1;            //隐式调用拷贝构造函数
    CTest1 d1(b1);            //显示调用拷贝构造函数
    CTest2 a2(2);            //显示调用构造函数
    CTest2 b2 = 2;            //隐式调用构造函数,编译错误
    CTest2 c2 = a2;            //隐式调用拷贝构造函数,编译错误
    CTest2 d2(b2);            //显示调用拷贝构造函数
    return 0;
}

如第26行代码所示,是直接隐式调用构造函数,创建对象b1,不要误理解为是先隐式调用构造函数创建临时对象,将调用拷贝构造函数,以临时对象创建对象b1.

如第31行代码所示,explicit对拷贝构造函数也会限制作用,将会阻隐式拷贝构造函数的调用

将拷贝构造函数声明为explicit,则会阻止隐式拷贝构造函数的调用.隐式拷贝构造函数的调用主要发生在三个点:

1.一个对象作为函数参数,以值传递的方式传入函数体

2.一个对象作为函数返回值,以值传递的方式从函数返回

3.以AAA = xxx的方式创建对象AAA,xxx为与AAA为同类型的对象.

因而,将拷贝构造函数声明成explicit并不是良好的设计,一般只将有单个参数的constructor声明为explicit,而copy constructor不要声明为explicit.

参考资料:http://baike.baidu.com/view/2422253.htm


猜你喜欢

转载自blog.csdn.net/dabenxiong_1/article/details/47981845