C++ Explicit关键字详细解析

时间:2021-12-07 06:18:11

explicit关键字用来修饰类的构造函数,表明构造函数是显示的,相对的是implicit关键字。
首先这个关键字只能用在类内部的构造函数声明上,而不能用在类外部的函数定义上,它的作用是不能进行隐式转换。

复制代码 代码如下:

    class gxgExplicit  //没有关键字explicit的类
    {
      public:
          int _size;
         gxgExplicit(int size)
        {
           _size = size;
        }
     };


下面是调用
   gxgExplicit gE1(24);     //这样是没有问题的
   gxgExplicit gE2 = 1;     //这样也是没有问题的
   gxgExplicit gE3;         //这样是不行的,没有默认构造函数
   gE1 = 2;                 //这样也是没有问题的
   gE2 = 3;                 //这样也是没有问题的
   gE2 = gE1;               //这样也是没有问题的

 

但是假如gxgExplicit修改为Stack,我们的_size代表的是堆栈的大小,那么调用的第二句就显得不伦不类,而且容易让人疑惑。这并不是可以让代码阅读者明白和接受的形式,虽然它是合法的(编译器可以通过编译)。这是因为编译器默认情况下有隐式转换的功能,你输入gE2 = 1就编译成同第一句相同的结果。所以,explicit就派上了用场。修改代码为:

复制代码 代码如下:

class gxgExplicit
    {
      public:
           int _size;
           explicit gxgExplicit(int size)
           {
             _size = size;
           }
    };


继续上面的调用:
   gxgExplicit gE1(24);     //这样是没有问题的
   gxgExplicit gE2 = 1;     //这样是不行的,关键字取消了隐式转换
   gxgExplicit gE3;         //这样是不行的,没有默认构造函数
   gE1 = 2;                 //这样是不行的,关键字取消了隐式转换
   gE2 = 3;                 //这样是不行的,关键字取消了隐式转换
   gE2 = gE1;               //这样是不行的,关键字取消了隐式转换,除非类实现操作符“=”的重载。

 

这是编译器(vs2005)显示:cannot convert from 'int' to 'gxgExplicit'。
从这里也就看出这个关键字的作用是将编译器隐式转换的功能给屏蔽掉。
MSDN上有一个注意点描述了下面的事实,当构造函数参数超过两个时自动取消隐式转换。例如

复制代码 代码如下:

    class gxgExplicit
    {
      private:
         int _size;
         int _age;
      public:
          explicit gxgExplicit(int age,int size)
         {
            _age = age;
            _size = size;
         }
     };


这是有没有关键字效果是一样的。那就是相当于有这个关键字。
但是另外一种情况例外:其中只有一个必须输入的参数,其余的为有默认值的参数。

复制代码 代码如下:

class gxgExplicit
{
private:
   int _size;
   int _age;
public:
   explicit gxgExplicit(int age,int size = 0)
   {
      _age = age;
      _size = size;
   }
};
class gxgExplicit
{
private:
   int _size;
   int _age;
int _hight;
public:
   explicit gxgExplicit(int age,int size = 0)
   {
      _age = age;
      _size = size;
      _hight = hight;
   }
};