explicit关键字用于防止单参数构造函数的隐式类型转换,避免编译器自动调用构造函数导致语义错误,如MyString(int)被误用于整数转字符串;加上explicit后需显式构造对象,提升代码安全性。从C++11起,explicit还可用于转换运算符,如explicit operator bool(),允许在条件判断中使用而禁止隐式转为bool变量,推荐用于安全布尔转换。建议对所有单参数构造函数使用explicit,除非明确需要隐式转换,尤其当参数为原始类型或包装类型时,以防止误用。

在C++中,explicit关键字主要用于防止编译器进行隐式类型转换,特别是针对单参数构造函数。如果不加explicit,编译器可能会在你不期望的情况下自动调用构造函数进行类型转换,从而引发潜在的bug。
防止隐式构造函数调用
当一个类的构造函数只有一个参数(或多个参数但除了第一个外都有默认值)时,它可能被用来做隐式转换。explicit可以阻止这种行为。
例如:
class MyString {public:
MyString(int size) { /* 分配size大小的字符串空间 */ }
};
void printString(const MyString& s) { }
立即学习“C++免费学习笔记(深入)”;
int main() {
printString(10); // 隐式将int转为MyString,可能不是你想要的
return 0;
}
上面代码中,传入整数10会触发MyString(int)构造函数,创建一个临时MyString对象。这虽然能通过编译,但语义上不合理——10并不是一个字符串。
加上explicit后:
class MyString {public:
explicit MyString(int size) { }
};
// printString(10); // 编译错误:不能隐式转换
printString(MyString(10)); // 正确:显式构造
这样就能避免误用,强制程序员明确表达意图。
适用于转换操作符(C++11起)
从C++11开始,explicit也可以用于用户定义的类型转换运算符,防止隐式转换。
比如:
class BooleanWrapper {private:
bool value;
public:
explicit operator bool() const { return value; }
};
使用时:
BooleanWrapper bw;if (bw) { } // 允许:条件判断中允许explicit bool
// bool b = bw; // 错误:不允许隐式转换
bool b = static_cast
这是现代C++中安全布尔转换的推荐做法,避免了老式operator void*带来的问题。
何时使用explicit
建议对所有单参数构造函数都使用explicit,除非你明确希望支持隐式转换。
- 构造函数接受原始类型(如int、double)时,几乎都应该加explicit
- 包装类型(如智能指针、字符串包装器)应避免隐式转换以防误用
- 当你定义类型之间“自然”可转换时(如String可由const char*构造),仍可考虑不加explicit,但需谨慎评估
基本上就这些。explicit是一个小关键字,但对提升代码安全性很有帮助。











