explicit关键字用于禁止单参数构造函数(或带默认值的多参数构造函数)的隐式类型转换,提升代码安全性与可读性;C++11起也适用于显式转换运算符。

explicit 关键字用来修饰单参数构造函数(或多个参数但除第一个外都有默认值的构造函数),目的是禁止编译器自动进行隐式类型转换。它不改变函数功能,只限制调用方式——让构造行为必须显式写出,提升代码可读性和安全性。
防止意外的隐式转换
没有 explicit 时,C++ 允许用一个值“悄悄”构造对象,容易引发难以察觉的 bug。比如:
class String {
public:
String(int n) { /* 分配 n 字节内存 */ }
};
String s = 10; // 隐式转换:编译器自动调用 String(10),容易被忽略
加上 explicit 后:
class String {
public:
explicit String(int n) { /* ... */ }
};
String s = 10; // ❌ 编译错误!不允许隐式转换
String s(10); // ✅ 正确:显式构造
String s = String(10); // ✅ 也允许,因为是显式类型转换
只对单参数构造函数起作用
explicit 只影响能被单个参数触发的构造路径。以下情况它才生效:
立即学习“C++免费学习笔记(深入)”;
- 只有一个参数的构造函数
- 多个参数,但除第一个外都有默认值(如 String(const char*, int len = 0)
如果有两个及以上参数且无默认值(如 String(const char*, int)),explicit 没有意义,本来就不能隐式转换。
explicit 也适用于转换运算符(C++11 起)
C++11 允许给类型转换函数加 explicit,防止隐式转出:
class Number {
int val;
public:
explicit operator int() const { return val; }
};
Number n;
int x = n; // ❌ 错误:不能隐式转成 int
int y = (int)n; // ✅ 正确:强制转换
int z = static_cast(n); // ✅ 也行
什么时候该用 explicit?
基本原则:只要该构造函数的语义不是“等价于”源类型的自然转换,就加上 explicit。
- 用 int 构造容器(如 String(10) 表示容量)→ 加 explicit
- 用 const char* 构造字符串 → 通常不加,因为 "hello" → String 是合理隐式转换
- 智能指针从原始指针构造(如 unique_ptr
(T*))→ 加 explicit,避免 delete 意外丢失
标准库中绝大多数单参数构造函数(如 std::vector、std::thread、std::unique_ptr)都用了 explicit。
基本上就这些。explicit 不复杂,但容易忽略——加在该加的地方,能帮你提前拦住不少隐蔽问题。









