C++ 显示关键字
- explicit 修饰构造函数时,可以防止隐式转换和复制初始化
- explicit 修饰转换函数时,可以防止隐式转换,但 按语境转换 除外
struct A{A(int) { }operator bool() const { return true; }};struct B{explicit B(int) {}explicit operator bool() const { return true; }};void doA(A a) {}void doB(B b) {}int main(){A a1(1); // OK:直接初始化A a2 = 1; // OK:复制初始化A a3{ 1 }; // OK:直接列表初始化A a4 = { 1 }; // OK:复制列表初始化A a5 = (A)1; // OK:允许 static_cast 的显式转换doA(1); // OK:允许从 int 到 A 的隐式转换if (a1); // OK:使用转换函数 A::operator bool() 的从 A 到 bool 的隐式转换bool a6(a1); // OK:使用转换函数 A::operator bool() 的从 A 到 bool 的隐式转换bool a7 = a1; // OK:使用转换函数 A::operator bool() 的从 A 到 bool 的隐式转换bool a8 = static_cast<bool>(a1); // OK :static_cast 进行直接初始化B b1(1); // OK:直接初始化B b2 = 1; // 错误:被 explicit 修饰构造函数的对象不可以复制初始化B b3{ 1 }; // OK:直接列表初始化B b4 = { 1 }; // 错误:被 explicit 修饰构造函数的对象不可以复制列表初始化B b5 = (B)1; // OK:允许 static_cast 的显式转换doB(1); // 错误:被 explicit 修饰构造函数的对象不可以从 int 到 B 的隐式转换if (b1); // OK:被 explicit 修饰转换函数 B::operator bool() 的对象可以从 B 到 bool 的按语境转换bool b6(b1); // OK:被 explicit 修饰转换函数 B::operator bool() 的对象可以从 B 到 bool 的按语境转换bool b7 = b1; // 错误:被 explicit 修饰转换函数 B::operator bool() 的对象不可以隐式转换bool b8 = static_cast<bool>(b1); // OK:static_cast 进行直接初始化return 0;}
通过explicit来防止隐式转换
#include <iostream>
struct TestClass {
template<typename T>
TestClass(const T &t) {
std::cout << "Constructed a TestClass " << t << std::endl;
}
};
void TakeATestClass(const TestClass &t) { }
int main() {
TakeATestClass("Bob");
}
对于上面的例子,尽管最终可能会通过编译器的编译,但是按照C++的原则“避免提供隐式转换”,对原有的类进行如下的修改可以防止隐式的转换。
struct TestClass {
template<typename T>
explicit TestClass(const T &t) {
std::cout << "Constructed a TestClass " << t << std::endl;
}
};
