C++ explicit关键字详解

简介:

单参数构造函数


单参数构造函数作为隐含的类型转换符号


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

  • 用于构建单参数的类对象
  • 隐含的类型转换操作符.

例如一个类A的构造函数A(int i)就是,既可以用来作为构造器,又可以实现隐式转换A a=1;因为1可以通过构造函数A(int i)转换为一个类A的对象。

#include <iostream>

class Base
{
public :
    Base(const char data)
    {
        std::cout <<"constructor..." <<std::endl;
        this->m_data = data;
    }

protected :

    char m_data;
};



int main(void)
{
    Base base1('a');        // 用于构建单参数的类对象
    Base base2 = 'b';     // 隐含的类型转换操作符

    return 0;
}

这种方法看起来很方便,但是有时这并不是我们想要的,但是由此可能引入一些其他问题。

由此带来的问题


比如如下的代码就可以正常运行

Base base = 10000;  //   由于10000可以被转成char类型

因此C++提供关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换发生.
特别对于如下代码

#include <iostream>


#include <cstring>


class Base
{
public :
    Base(const char *str)
    {
        std::cout <<"constructor..." <<std::endl;

        this->m_length = strlen(str);
        this->m_cstr = new char[this->m_length + 1];

        strcpy(this->m_cstr, str);
    }

    Base(const Base &base)
    {
        this->m_length = base.m_length;
        this->m_cstr = new char[this->m_length + 1];

        strcpy(this->m_cstr, base.m_cstr);
    }

    ~Base()
    {
        delete this->m_cstr;
    }

protected :

    char *m_cstr;
    int  m_length;
};



int main(void)
{
    Base base = (char *)10000;     // 隐含的类型转换操作符

    return 0;
}

这里写图片描述
声明为explicit的构造函数不能在隐式转换中使用,只能显示调用,去构造一个类对象。

这里写图片描述

拷贝构造函数copy construct


Copy constructor也是同样的,如果Copy constructor被声明为explicit,则这个类对象不能用于传参和函数返回值。但是仍然可以直接调用。

拷贝构造函数的调用时机


我们都知道,在C++中,拷贝构造函数的调用时机有如下三种情况

  1. 当用类的一个对象去初始化该类的另一个对象时
  2. 如果函数的形参是类的对象,调用函数,进行形参和实参结合时
  3. 如果函数的返回值是类的对象,函数执行完成返回调用者时
#include <iostream>

class Base
{
public :
    Base( )
    {
        //std::cout <<"default constructor..." <<std::endl;
        this->m_data = '\0';
    }

    Base(char data)
    {
        //std::cout <<"simple constructor..." <<std::endl;
        this->m_data = data;
    }

    Base(const Base &base)
    {
        std::cout <<"copy constructor..." <<std::endl;
        this->m_data = base.m_data;
    }

protected :

    char m_data;
};


//  参数为Base对象
void Func(const Base base)
{
}

//  返回值为base对象
Base Func( )
{
    Base base;

    return base;
}

void Copy()
{
    Base b1;
    Base b2 = b1;

}

int main(void)
{
    Base base(10);

    std::cout  <<std::endl <<"参数为Base对象..." <<std::endl;
    Func(base);     //  优化未优化情况下均调用1次拷贝构造函数

    std::cout  <<std::endl <<"返回值为base对象[可被优化]..." <<std::endl;
    Func();     //  优化情况下调用0次, 未优化情况下调用1次[函数返回时调用]

    Base b = Func();     //  优化情况下调用0次, 未优化情况下调用2次[函数返回时调用 + 初始化类型b时调用]


    std::cout  <<std::endl <<"用base对象初始化另外一个对象..." <<std::endl;
    Copy();     //  优化未优化情况下均调用1次拷贝构造函数

    return 0;
}

但是其实很多现代编译器的返回值优化技巧,返回值时的情况会被优化掉,但是这个并不是我们今天讨论的重点。

具体可参见

RVO-编译器返回值优化

在使用GNU/g++编译器时可以使用”-fno-elide-constructors”选项来强制g++总是调用copy构造函数,即使在用临时对象初始化另一个同类型对象的时候(主要是返回值为)

explicit对拷贝构造函数也会限制作用,将会阻止隐式拷贝构造函数的调用。
explicit对拷贝构造函数也会限制作用,将会阻隐式拷贝构造函数的调用.

将拷贝构造函数声明为explicit,则会阻止隐式拷贝构造函数的调用。

explicit关键字对拷贝构造函数的限制


这时候我们可以将参数传递或者函数返回修改为引用的方法,而初始化的时候采用显式调用拷贝构造函数。

#include <iostream>

class Base
{
public :
    Base( )
    {
        //std::cout <<"default constructor..." <<std::endl;
        this->m_data = '\0';
    }

    explicit Base(char data)
    {
        //std::cout <<"simple constructor..." <<std::endl;
        this->m_data = data;
    }

    explicit Base(const Base &base)
    {
        std::cout <<"copy constructor..." <<std::endl;
        this->m_data = base.m_data;
    }

protected :

    char m_data;
};


//  参数为Base对象
void Func(const Base& base)     //  传入引用对象
{
}

//  返回值为base对象
Base& Func( )
{
    Base base;

    return base;            //  返回引用对象
}

void Copy()
{
    Base b1;
    Base b2(b1);            //  Base b2 = b1显示调用拷贝构造函数

}

int main(void)
{
    Base base(10);

    std::cout  <<std::endl <<"参数为Base对象..." <<std::endl;
    Func(base);     //  优化未优化情况下均调用1次拷贝构造函数

    std::cout  <<std::endl <<"返回值为base对象[可被优化]..." <<std::endl;
    Func();     //  优化情况下调用0次, 未优化情况下调用1次[函数返回时调用]

    //Base b(Func());     返回局部对象


    std::cout  <<std::endl <<"用base对象初始化另外一个对象..." <<std::endl;
    Copy();     //  优化未优化情况下均调用1次拷贝构造函数

    return 0;
}

总结


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

  • 用于构建单参数的类对象
  • 隐含的类型转换操作符

声明为explicit的构造函数不能在隐式转换中使用,只能显示调用,去构造一个类对象。

explicit关键字只对有一个参数的类构造函数有效, 如果类构造函数参数大于或等于两个时, 是不会产生隐式转换的, 所以explicit关键字也就无效了

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



转载:http://blog.csdn.net/gatieme/article/details/50867575


目录
相关文章
|
1月前
|
存储 安全 算法
【C/C++ 关键字 函数说明符 】C++ noexcept 关键字(指定某个函数不抛出异常)
【C/C++ 关键字 函数说明符 】C++ noexcept 关键字(指定某个函数不抛出异常)
26 0
|
1月前
|
设计模式 算法 安全
【C/C++ 关键字 函数说明符 】C++ final关键字(修饰成员函数无法被子类重写覆盖)
【C/C++ 关键字 函数说明符 】C++ final关键字(修饰成员函数无法被子类重写覆盖)
39 1
|
1月前
|
算法 安全 编译器
【C++ 关键字 override】C++ 重写关键字override(强制编译器检查该函数是否覆盖已存在的虚函数)
【C++ 关键字 override】C++ 重写关键字override(强制编译器检查该函数是否覆盖已存在的虚函数)
27 0
|
1月前
|
算法 Java 编译器
【C++ 关键字 virtual 】C++ virtual 关键字(将成员函数声明为虚函数实现多态
【C++ 关键字 virtual 】C++ virtual 关键字(将成员函数声明为虚函数实现多态
25 0
|
1月前
|
存储 缓存 安全
【C/C++ 关键字 存储类说明符 】 线程局部变量的魔法:C++ 中 thread_local的用法
【C/C++ 关键字 存储类说明符 】 线程局部变量的魔法:C++ 中 thread_local的用法
33 0
|
1月前
|
存储 安全 编译器
【C++ 关键字 类型限定符 】揭秘C++编程中的神秘元素:深入了解volatile关键字的强大作用
【C++ 关键字 类型限定符 】揭秘C++编程中的神秘元素:深入了解volatile关键字的强大作用
21 0
|
16天前
|
编译器 C语言 C++
【C++】C++入门第一课(c++关键字 | 命名空间 | c++输入输出 | 缺省参数)
【C++】C++入门第一课(c++关键字 | 命名空间 | c++输入输出 | 缺省参数)
|
16天前
|
存储 程序员 编译器
C++注释、变量、常量、关键字、标识符、输入输出
C++注释、变量、常量、关键字、标识符、输入输出
|
18天前
|
编译器 C语言 C++
【C++的奇迹之旅(二)】C++关键字&&命名空间使用的三种方式&&C++输入&输出&&命名空间std的使用惯例
【C++的奇迹之旅(二)】C++关键字&&命名空间使用的三种方式&&C++输入&输出&&命名空间std的使用惯例
|
1月前
|
存储 算法 编译器
【C++ 关键字 static_assert 相关问题】C++ 关于静态断言的编译问题 ,深入了解静态断言
【C++ 关键字 static_assert 相关问题】C++ 关于静态断言的编译问题 ,深入了解静态断言
29 0

热门文章

最新文章