explicit 关键字用于防止构造函数进行隐式类型转换,避免意外行为。当类的构造函数只有一个参数时,编译器会默认允许隐式转换,例如将 int 自动转为 MyString 对象,可能导致逻辑错误。使用 explicit 后,只能通过显式调用构造函数创建对象,禁止了这种自动转换。从 C ++11 起,explicit 也适用于多参数构造函数,控制统一初始化语法下的隐式转换。建议在单参数构造函数中普遍使用 explicit,除非确实需要隐式转换,常见于包装类、数值类和资源管理类,以增强类型安全性和代码清晰性。

在 C ++ 中,explicit关键字主要用于修饰单参数构造函数(或能以单参数调用的多参数构造函数),防止编译器进行 隐式类型转换。如果不使用explicit,编译器可能会在你不察觉的情况下调用构造函数进行自动转换,从而引发意外行为。
为什么 需要 explicit?
当一个类的构造函数只有一个参数时,它会默认成为一个“转换构造函数”。这意味着编译器可以自动将该参数类型的值转换为这个类的对象。
例如:
class MyString {public: MyString(int size) {// 假设根据整数分配字符串缓冲区} }; void printString(const MyString& str) {// 打印字符串} printString(10); // 隐式转换:int → MyString,可能不是你想要的!
上面代码中,printString(10) 能够通过编译,因为编译器自动调用了 MyString(int) 构造函数。这可能导致逻辑错误,比如误把大小当作字符串对象传入。
立即学习“C++ 免费学习笔记(深入)”;
使用 explicit 阻止 隐式转换
加上 explicit 关键字后,构造函数不能再被用于隐式转换,只能显式调用。
class MyString {public: explicit MyString(int size) {// 构造逻辑} }; // printString(10); // 错误:无法隐式转换 int → MyString printString(MyString(10)); // 正确:显式创建对象 printString{20}; // 错误:列表初始化也不允许隐式转换
此时,任何试图隐式转换的行为都会导致 编译错误,从而避免了潜在的误用。
explicit 在多参数构造函数中的使用(C++11 起)
C++11 引入了对多参数构造函数使用 explicit 的支持,尤其是在使用统一初始化语法(花括号 {})时。
class Point {public: explicit Point(int x, int y) : x_(x), y_(y) {} private: int x_, y_;}; // Point p = {1, 2}; // 错误:explicit 禁止这种隐式列表初始化 Point p{1, 2}; // 正确:显式初始化
这样可以控制哪些构造方式是允许的,增强类型安全性。
何时使用 explicit?
建议在大多数单参数构造函数前加上 explicit,除非你明确希望支持隐式转换。常见场景包括:
- 包装基本类型的类(如 String、Array、SmartPointer)
- 数值类型转换类(如 Complex、Rational)
- 资源管理类(如 FileHandle、LockGuard)
总之,explicit 是一种防御性编程手段,帮助你写出更安全、更清晰的代码。
基本上就这些,加上 explicit 不复杂但容易忽略,养成习惯能避免不少坑。
以上就是 C ++ 如何使用 explicit 关键字_C++ 防止构造函数的






























