在C++中,根据默认的规定,在定义只有一个参数的构造函数时同时也定义了一个隐式转换:将构造函数的参数对应的数据类型的数据转化为该类的对象,如下面代码所示:
class CMyString { CMyString(const char * var) ;//C风格字符串var作为初始化值 ....... }; CMyString myString = "EXPLICIT example."; //隐式转换,相当于CMyString myString = CMyString("EXPLICIT example.");
但是这种隐式转换会造成歧义,例如:
class CMyString { CMyString(int byteLen);//分配字符串byteLen个字节 CMyString(const char *var);//用C风格字符串var作为初始化值 ...... };
下面的两种初始化是正常的:
CMyString firstStr(10); //分配10个字节的空字符串 CMyString secStr = CMyString(10); //分配10个字节的空字符串
但是接下来的两种初始化就会引起歧义:
CMyString thirdStr = 10; //也是分配10个字节的空字符串 CMyString fourthStr = 'a'; //注意!这里分配 int('a') 个字节的空字符串
thirdStr和fourthStr分别把 int型和 char型,隐式转换成了分配相应字节的空字符串,这容易引起歧义。
同样,下面的情况也会引起隐式转换,造成歧义:
void createStr(CMyString str); //调用函数createStr createStr(10);
当调用createStr时,会将参数10隐式转换为CMyString类型,这也会引起歧义。
为了避免这种歧义的发生,可以使用关键字explicit声明显示的转换:
class CMyString { explicit CMyString(int byteLen);//分配字符串byteLen个字节 CMyString(const char *var);//用C风格字符串var作为初始化值 ...... };
下面的两种写法依然正确:
CMyString firstStr(10); //分配10个字节的空字符串 CMyString secStr = CMyString(10); //分配10个字节的空字符串
但是,下面的两种写法就不正确了:
CMyString thirdStr = 10; //编译不通过 CMyString fourthStr = 'a'; //编译不通过
同样,调用函数createStr时的隐式转换也不被支持了:
//调用函数createStr createStr(10); //编译不通过
总结:关键字explicit只对构造函数起作用,用来抑制隐式转换。
转自:http://www.cnblogs.com/cutepig/archive/2009/01/14/1375917.html