explicit关键字用于防止构造函数的隐式类型转换。1. 它修饰单参数构造函数,避免编译器自动将参数类型转换为类类型,如MyString(int)被误用;2. 加explicit后,必须显式构造对象,禁止printString(10)这类调用,提升安全性和可读性;3. C++11起支持多参数构造函数,阻止花括号语法的隐式转换,如Point(int x, int y);4. 建议所有单参数或默认参数导致单参数的构造函数使用explicit,防止意外构造,增强类型安全。

在C++中,explicit关键字主要用于修饰单参数构造函数(或能通过默认参数变成单参数的构造函数),用来防止编译器进行隐式类型转换。如果不使用explicit,编译器可能会在你不注意的情况下自动调用构造函数进行类型转换,这有时会导致意想不到的行为。
为什么需要explicit?
考虑以下情况:
class MyString {
public:
MyString(int size) {
// 假设这里分配size个字符的空间
}
};
void printString(const MyString& str) {
// 打印字符串
}
// 调用
printString(10); // 这行代码会编译通过吗?
如果没有explicit,上面的代码是合法的。编译器会自动将整数10隐式转换为MyString对象,调用MyString(int)构造函数。这可能不是你想要的结果——你本意可能是传一个字符串,结果却传了个数字,程序还能运行,但逻辑出错。
使用explicit避免隐式转换
加上explicit后,上述隐式转换就会被禁止:
立即学习“C++免费学习笔记(深入)”;
class MyString {
public:
explicit MyString(int size) {
// 构造逻辑
}
};
printString(10); // 编译错误!不允许隐式转换
printString(MyString(10)); // 正确:显式创建对象
这时,必须显式地创建MyString对象,才能传递给函数,提高了代码的安全性和可读性。
explicit适用于多个参数的情况(C++11起)
在C++11之前,explicit只能用于单参数构造函数。但从C++11开始,explicit也可以用于多参数构造函数,主要是为了防止通过统一初始化语法(花括号{})发生隐式转换。
class Point {
public:
explicit Point(int x, int y) {
// 初始化坐标
}
};
void drawPoint(const Point& p) { }
drawPoint({1, 2}); // 错误:explicit禁止隐式转换
drawPoint(Point{1, 2}); // 正确:显式构造
虽然多参数构造函数本来就不会发生传统意义上的隐式转换,但explicit可以阻止使用{ }语法的隐式转换,进一步增强类型安全。
什么时候该用explicit?
一般建议:只要构造函数只接受一个参数(或能变成一个参数),并且你不希望它被用于隐式转换,就应声明为explicit。
- 包装类型(如智能指针、字符串类)的构造函数通常应标记为explicit
- 避免“意外构造”带来的逻辑错误
- 提高接口清晰度:用户必须明确表达意图
基本上就这些。explicit不是一个复杂的关键字,但它在防止隐式转换引发的bug方面非常有用。合理使用,能让代码更安全、更清晰。











