C++ 语言可以定义如何将其他类型的对象隐式转换为我们的类类型, 或将我们的类类型的对象隐式转换为其他类型。
下面为类类型的隐式转换的示例代码:
#include
using namespace std;
class A
{
public:
A(int temp) //普通构造函数
{
a = temp;
cout << 普通构造函数: a= << a << endl;
}
A(const A &temp) //拷贝构造函数
{
a = temp.a;
cout << 拷贝构造函数: a = << a << endl;
}
private:
int a;
};
int main()
{
cout << 显示调用: << endl;
A a(250); //显式调用调用普通构造函数,ok
A b(a); //显式调用调用拷贝构造函数,ok
cout <<
隐式转换: << endl;
A c=222; //隐式转换成调用普通构造函数,ok
A d=c; //隐式转换成调用拷贝构造函数,ok
return 0;
}
编译运行结果如下:
如果要避免这种自动转换的功能,我们该怎么做呢?C++提供了关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换的发生。
c++中的explicit关键字抑制由构造函数定义的隐式转换。explicit关键字只能用于类内部的构造函数声明上。在类的定义体外部所做的定义上不要加explicit关键字。
示例代码如下:
#include
using namespace std;
class A
{
public:
explicit A(int temp) //普通构造函数,被声明为explicit(显式)
{
a = temp;
cout << 普通构造函数: a= << a << endl;
}
explicit A(const A &temp) //拷贝构造函数,被声明为explicit(显式)
{
a = temp.a;
cout << 拷贝构造函数: a = << a << endl;
}
private:
int a;
};
int main()
{
cout << 显示调用: << endl;
A a(250); //显式调用调用普通构造函数,ok
A b(a); //显式调用调用拷贝构造函数,ok
//explicit构造函数只能被显式调用
A c=222; //不能通过隐式转换,error
A d=c; //不能通过隐式转换,error
return 0;
}
编译时会报错:
普通构造函数能够被隐式调用。而explicit构造函数只能被显式调用。
通常,除非有明显的理由想要定义隐式转换,否则,单形参构造函数应该为 explicit。将构造函数设置为explicit可以避免错误,并且当转换有用时,用户可以显式地构造对象。