class CExample
{
public:
CExample(void);
explicit CExample(int iFirst, int iSecond = 4);
~CExample(void);
public:
int m_iFirst;
int m_iSecond;
};
//Example.cpp
#include "Example.h"
CExample::CExample(void)
: m_iFirst(0)
{
}
CExample::CExample(int iFirst, int iSecond)
:m_iFirst(iFirst), m_iSecond(iSecond)
{
}
CExample::~CExample(void)
{
}
在没有关键字
explicit
的情况下
int main(int argc, char* argv[])
{
CExample objOne; //调用没有参数的构造函数
CExample objTwo(12, 12); //调用有两个参数的构造函数
CExample objThree(12); //同上,可以传一个参数是因为该构造函数的第二个参数有默认值
CExample objFour = 12; //执行了隐式转换,等价于CExample temp(12);objFour(temp);
//注意这个地方调用了
编译器为我们提供的默认复制构造函数
return 0;
}
但是在含有explicit关键字的情况下,
CExample objFour = 12;
这样的语句将会出现编译错误。
因为关键字explicit的作用就是阻止隐式转换的发生,防止发生不可预知的错误。
对于某些情况下隐式转换可能给我们带来方便,但在大部分情况中,隐式转换却容易导致错误(不是语法错误,编译器不会报错)。
隐式转换总是在我们没有察觉的情况下悄悄发生,除非有心所为,隐式转换常常是我们所不希望发生的。
通过将构造函数声明为explicit(显式)的方式可以抑制隐式转换。也就是说,explicit构造函数必须显式调用。
发生隐式转换,除非有心利用,隐式转换常常带来程序逻辑的错误,而且这种错误一旦发生是很难察觉的。
原则上应该在所有的构造函数前加explicit关键字,当你有心利用隐式转换的时候再去解除explicit,这样可以大大减少错误的发生。