先来看一段代码:
#include <iostream>
using namespace std;
class Test{
int a;
public:
Test():a(0){cout<<"void";}
explicit Test(int i):a(i){cout<<"int";}
Test(short s):a(s){cout<<"short";}
Test& operator=(int n){a=n;cout<<"operator=";}
};
int main()
{
int n;
Test a = n;
return 0;
}
首先先分析下会输出什么,再往下看。
这段代码会输出 short;
在C++中,只有一个参数的构造函数(或者除了第一个参数外其余参数都有默认值的多参构造函数),
承担了两个角色:1.是构造器,2.是默认且隐含的类型转换操作符。
第一个比较好理解,第二个作用其实意思就是当写下如:
Object obj = 10;
这样的代码时 会调用obj的只有一个参数的构造函数。
但这样有时会产生歧义,如:
class String{
public:
String(int n); // 分配n个字节
String(const char* p); // 使用c风格字符串初始化值
};
int main()
{
String s = 'a';
return 0;
}
这里会调用第一个构造函数分配'a'(97)个大小的字节,但是这样比较迷惑,有人觉着我是在用c风格字符串初始化,
怎么变成分配字节了。
于是引入了explicit关键字来消除这种迷惑或误解。
关键字explicit的作用是:阻止不应该允许的经过转换构造函数进行的隐式转换的发生。声明为explicit的构造函数不能
在隐式转换中使用。
也就是封杀了一个构造函数的第二个功能。
那么下面来看最前面的那个问题。
首先无论如何这里不会调用赋值运算符。这里不清楚为什么的要去好好补补。
main函数中肯定是要进行隐式转换调用构造函数,所以不会是第一个无参构造函数。
所以就是第二三个构造函数了,按理 参数是int类型 应该调用第二个输出int。
但是int对应的那个构造函数被声明为了explicit,拒绝隐式转换,所以也不能调用这个。
接下来只能int参数在做个隐式转换转换为short类型,这样就行得通了。
此时会调用第三个构造函数,所以输出为short。
总结:explicit只对一个参数的构造函数起作用。用来抑制隐式转换。
题目来源:牛客网https://www.nowcoder.com/profile/4765071/myFollowings/detail/4633330