C++构造函数中explicit详解

 更新时间:2025年12月03日 11:07:26   作者:湫兮之风  
explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以了解一下

在 C++ 编程中,构造函数是类的核心部分之一。我们常常使用构造函数来初始化对象。但是,如果不加限制,某些构造函数可能会被 隐式调用,从而带来一些意料之外的行为。

为了解决这个问题,C++ 提供了 explicit 关键字。

1. 什么是explicit

在 C++ 中,explicit 关键字用于修饰 单参数构造函数可以看作单参数的构造函数,阻止编译器进行 隐式类型转换拷贝初始化

  • 默认情况下,单参数构造函数既可以显式调用,也可以被编译器用来进行隐式类型转换。
  • explicit 告诉编译器:这个构造函数只能显式调用,不能用于隐式转换。

2. 隐式转换的问题

来看一个例子:

#include <iostream>
using namespace std;

class Fraction {
private:
    int numerator;
    int denominator;

public:
    Fraction(int num, int den = 1) : numerator(num), denominator(den) {}
    void print() const {
        cout << numerator << "/" << denominator << endl;
    }
};

int main() {
    Fraction f1 = 5;  // 隐式调用 Fraction(5, 1)
    f1.print();       // 输出:5/1
}

在上面的例子中:

  • Fraction f1 = 5; 本质上是调用 Fraction(5, 1),因为编译器允许用 int 隐式转换成 Fraction
  • 虽然看似方便,但有时会带来 不可控的隐式转换,导致逻辑错误或二义性。

3.explicit的使用示例

基本用法

如果我们在构造函数前加上 explicit

class Fraction {
private:
    int numerator;
    int denominator;

public:
    explicit Fraction(int num, int den = 1) : numerator(num), denominator(den) {}
    void print() const {
        cout << numerator << "/" << denominator << endl;
    }
};

int main() {
    Fraction f1(5);     // ✅ 显式调用,可以
    // Fraction f2 = 5; // ❌ 编译错误,不能隐式转换
}
  • Fraction f1(5); 依然可以显式调用。
  • Fraction f2 = 5; 会报错,因为 explicit 禁止了隐式转换。

多参数构造函数

有时构造函数有多个参数,但如果除第一个外的参数都有默认值,它依然算作 单参数构造函数,也可能引发隐式转换。

class Fraction {
public:
    explicit Fraction(int num, int den = 1) { /* ... */ }
};

这里如果没有 explicit,表达式 Fraction f = 5; 依然会成立。

4. C++11 之后的扩展

(1)explicit用于转换运算符

在 C++11 之前,类的类型转换函数(比如 operator bool)会允许隐式转换:

class Test {
public:
    operator bool() const { return true; }
};

int main() {
    Test t;
    if (t) {  // 隐式调用 operator bool()
        cout << "True" << endl;
    }
}

但有时我们并不希望这种隐式转换。C++11 允许写成:

class Test {
public:
    explicit operator bool() const { return true; }
};

int main() {
    Test t;
    // if (t) { } // ❌ 错误,不能隐式转换
    if (static_cast<bool>(t)) {  // ✅ 必须显式转换
        cout << "True" << endl;
    }
}

(2) C++20 的explicit(bool)

C++20 引入了更灵活的语法:explicit(bool)
这让我们可以根据编译期常量决定是否允许隐式调用。

struct A {
    explicit(true) A(int) {}   // 永远显式
    explicit(false) A(double) {} // 永远允许隐式
};

这种写法在模板编程中很有用。

5. 最佳实践

  1. 几乎总是给单参数构造函数加 explicit
    这样可以避免隐式转换带来的混乱,除非你确实需要这种转换。

  2. 转换运算符应当尽量显式
    尤其是 operator bool,因为隐式转换到 bool 可能导致奇怪的条件判断。

  3. 允许隐式转换的场景
    如果你的类本质上就是包装某个类型(比如 string_view 可以从 const char* 隐式转换),那么允许隐式转换可以让使用更加自然。

总结

  • explicit 的主要作用:防止构造函数或转换运算符被隐式调用。
  • 在单参数构造函数和转换运算符中使用最为常见。
  • 自 C++11 起,还能用于 operator bool;C++20 引入 explicit(bool),进一步增强灵活性。
  • 最佳实践:默认加上 explicit,除非你有充分理由允许隐式转换。
  • 关键字explicit只对一个实参的构造函数有效,需要多个实参的构造函数不能用于执行隐式转换,所以无需将这些构造函数指定为explicit的。
  • 只能在类内声明构造函数时使用explicit关键字。

补充

  • 接受一个单参数的const char*的string构造函数不是explicit的
  • 接受一个容量参数的vector构造函数是explicit的

到此这篇关于C++构造函数中explicit详解的文章就介绍到这了,更多相关C++构造函数explicit内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 如何C++使用模板特化功能

    如何C++使用模板特化功能

    这篇文章主要介绍了如何C++使用模板特化功能,通过定义模板,使得函数或者类不依赖于特定的类型,这样大幅提升了代码的复用性。更多详细内容需要的小伙伴可以参考一下
    2022-03-03
  • C++实现宿舍管理查询系统

    C++实现宿舍管理查询系统

    这篇文章主要为大家详细介绍了C++实现宿舍管理查询系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • 深入解析C++编程中的纯虚函数和抽象类

    深入解析C++编程中的纯虚函数和抽象类

    这篇文章主要介绍了深入解析C++编程中的纯虚函数和抽象类,是C++入门学习中的基础知识,需要的朋友可以参考下
    2015-09-09
  • C++实现LeetCode(201.数字范围位相与)

    C++实现LeetCode(201.数字范围位相与)

    这篇文章主要介绍了C++实现LeetCode(201.数字范围位相与),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • C++中函数匹配机制详解

    C++中函数匹配机制详解

    大家好,本篇文章主要讲的是C++中函数匹配机制详解,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下
    2022-02-02
  • C++中回调函数(CallBack)的用法分析

    C++中回调函数(CallBack)的用法分析

    这篇文章主要介绍了C++中回调函数(CallBack)的用法,较为详细的分析了C++中回调函数(CallBack)的原理并以实例形式总结了其具体用法,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-01-01
  • C++中的重载、覆盖、隐藏介绍

    C++中的重载、覆盖、隐藏介绍

    这篇文章主要介绍了C++中的重载、覆盖、隐藏介绍,需要的朋友可以参考下
    2015-04-04
  • C语言数据结构之二叉链表创建二叉树

    C语言数据结构之二叉链表创建二叉树

    这篇文章主要介绍了C语言数据结构之 二叉链表创建二叉树,下文我们为了更方便的使用二叉树结构体,可以使用 typedef 对结构体进行命名,具体内容需要的小伙伴可以参考一下
    2022-02-02
  • C++设计模式之工厂方法模式的实现及优缺点

    C++设计模式之工厂方法模式的实现及优缺点

    工厂方法模式是一个创建型设计模式,通过定义一个创建对象的接口,让其子类决定实例化哪一个工厂类,这篇文章主要给大家介绍了关于C++设计模式之工厂方法模式的实现及优缺点,需要的朋友可以参考下
    2021-06-06
  • C++实现单例模式的方法

    C++实现单例模式的方法

    这篇文章主要为大家介绍了C++实现单例模式的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2021-12-12

最新评论