【正确答案】
【答案解析】在C++中,如下声明是合法的。
class String
{
String(const char* p);
...
}
String s1="hello";
上例中,String s1="hello"会执行隐式转换,等价于String s1=String("hello")。为了避免这种情况的发生,C++引入了关键字explicit,它可以阻止不应该允许的经过转换构造函数进行的隐式转换的发生,声明为explicit的构造函数不能在隐式转换中使用。
在C++中,一个参数的构造函数(或者除了第一个参数外其余参数都有默认值的多参构造函数)一般具备两个功能:构造器和默认且隐含的类型转换操作符。所以,当AAA=XXX,恰好XXX的类型正好是AAA单参数构造器的参数类型,这时候编译器就自动调用这个构造器,创建一个AAA的对象。而在某些情况下,却违背了程序员的本意。此时就要在这个构造器前面加上explicit修饰,指定这个构造器只能被明确地调用、使用,不能作为类型转换操作符被隐含地使用。
程序代码如下:
class Test1
{
public:
Test1(int n){num=n;} ∥普通构造函数
private:
int num;
};
class Test2
{
public:
explicit Test2(int n){num=n;) ∥explicit(显式)构造函数
private:
int num;
};
int main()
{
Test1 t1=12; ∥隐式调用其构造函数,成功
Test2 t2=12; ∥编译错误,不能隐式调用其构造函数
Test2 t3(12); ∥显示调用成功
return 0;
}
Test1的构造函数带一个int型的参数,Test1 t1=12会隐式转换成调用Test1的这个构造函数,而Test2的构造函数被声明为explicit(显式),这表示不能通过隐式转换来调用这个构造函数,因此Test2 t2=12会出现编译错误。普通构造函数能够被隐式调用,而explicit()构造函数只能被显式调用。