C++ explicit 用法总结

本文详细解释了C++中explicit关键字的作用及正确用法,强调了它不仅适用于单参数构造函数,还可以应用于所有参数均有默认值的多参数构造函数,以防隐式类型转换导致的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        关于explicit关键字的用法,网上有中说法是C++explicit关键字只能用于修饰只有一个参数的构造函数,其实这种说法是不准确的,有误导性。

        explicit关键字是为了防止构造函数的隐式转换的,理论上说,多参构造是不会发生隐式转换的,但是,如果,多参构造的参数都有默认值的话,那么隐式转换也会发生的,所以如果多参,且每个参数都有默认值的话,为了防止隐式转换也需要使用explicit关键字,什么是隐式转换,看下面例子


#include <iostream>

class point {

public:
    int x, y,z; 
   point(int x = 0, int y = 0) :x(x), y(y)
   {

   }

};
void printPoint(const point& p)
{
    std::cout << "x=" << p.x << " " << p.y << std::endl;
}
int main()
{
    printPoint(1);
    point p = 2;
    printPoint(p);
}

 输出结果如下,可以看到printpoint函数,正常输出 x=1 y=0,我们创建的p变量并赋值1 也正常打印了,并没有给我们报错,以上的代码示例就是c++给我们进行的隐式转换,而explicit关键字就是为了防止这种隐式转换发生,以免造成难以调试的问题

explicit关键字

指定构造函数为显示,不能用于隐式转换 和 复制转换

如果我们在我们的构造函数前加上explicit关键字,那么编译阶段就会报错,避免了运行期错误,在编码过程中能用尽用

Effective C++中的说法

被声明为explicit的构造函数通常比其 non-explicit 兄弟更受欢迎, 因为它们禁止编译器执行非预期 (往往也不被期望) 的类型转换. 除非我有一个好理由允许构造函数被用于隐式类型转换, 否则我会把它声明为explicit. 我鼓励你遵循相同的政策.

### 明确 `explicit` 关键字的作用 在 C++ 中,`explicit` 关键字用于修饰单参数构造函数以及转换运算符,其主要目的是防止隐式的类型转换和拷贝初始化带来的潜在错误[^2]。 当定义了一个类的构造函数时,默认情况下编译器会允许该构造函数被用来执行隐式类型转换。然而,在某些场景下这种行为可能会引发意外的结果或者难以察觉的 bug。通过将构造函数声明为 `explicit`,可以显式地禁用这些不必要的隐式转换操作[^3]。 #### 防止隐式类型转换的例子 下面展示如何利用 `explicit` 来控制对象创建过程中的自动类型推导: ```cpp class MyClass { public: explicit MyClass(int value) : m_value(value) {} // 使用 explicit 限定词 private: int m_value; }; void process(const MyClass& obj) {} int main() { // 下面这行代码会产生编译期错误,因为 MyClass 的构造函数标记为了 explicit //process(42); // 正确的方式是手动调用构造函数来实例化对象 process(MyClass(42)); } ``` 在这个例子中,如果去掉 `MyClass` 构造函数前的 `explicit` 定义,则可以直接传递整数给 `process()` 函数而无需显式构建 `MyClass` 对象。这样虽然方便了编码工作量减少,但也增加了程序逻辑上的模糊性和可能的风险[^4]。 #### 转换运算符的情况 除了应用于构造函数外,`explicit` 还能作用于用户自定义类型的转换运算符上,从而阻止不期望发生的隐式转型动作发生: ```cpp struct ExplicitConversion { operator bool() const { return true; } }; // 假设我们有一个只接受布尔型变量作为条件表达式的函数 bool check(bool condition); ExplicitConversion ec; if (check(ec)) { /* ... */ } // 如果没有指定 explicit 则此句合法并尝试转化ec到bool类型 // 加入 explicit 后则需改为如下形式才能正常运行 if (check(static_cast<bool>(ec))) {} ``` 以上情况表明即使存在有效的类型转换路径,只要目标类型涉及到了带有 `explicit` 属性的操作符成员方法,那么就必须采用强制造型手段(如 static_cast<>)来进行显示指明意图[^5]。 ### 总结 综上所述,合理运用 `explicit` 可以帮助开发者编写更加安全可靠的现代 C++ 应用程序,尤其是在设计复杂的数据结构或是实现库级组件的时候尤为重要[^6]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值