默认构造函数

默认构造函数

< cpp‎ | language

默认构造函数是能以无参数调用的构造函数(以空参数列表定义,或以为每个参数提供默认参数定义)。拥有公开默认构造函数的类型是可默认构造 (DefaultConstructible) 的。

语法

class_name ( ) ;(1)
class_name :: class_name ( ) body(2)
class_name() = delete ;(3)(C++11 起)
class_name() = default ;(4)(C++11 起)
class_name :: class_name ( ) = default ;(5)(C++11 起)

其中 class_name 必须指名当前类(或类模板的当前实例化),或在命名空间作用域或友元声明中声明时,必须是有限定类名。

解释

1) 默认构造函数的声明。

2) 默认构造函数的类体外定义。

3) 禁止编译器自动生成默认构造函数。

4) 显式强制编译器自动生成默认构造函数。

默认构造函数在默认初始化和值初始化时得到调用。

隐式声明的默认构造函数

若不对类类型(struct 、 class 或 union)提供任何用户声明的构造函数,则编译器将始终自动声明作为其类的 inline public 成员的默认构造函数。

若存在某个用户声明的构造函数,则用户仍可以关键词 default 强制编译器自动生成本可隐式声明的默认构造函数。

(C++11 起)

隐式声明(或在其首声明设为默认)的默认构造函数拥有描述于动态异常规定 (C++17 前)异常规定 (C++17 起)的异常规定。

隐式定义的默认构造函数

若隐式声明的默认构造函数未被定义为被删除,则若得到odr 使用,则它为编译器所定义(即生成函数体并编译),且它与拥有空体和空初始化器列表的用户定义构造函数有准确相同的效果。即它调用基类和此类的非静态成员的默认构造函数。若它满足 constexpr 构造函数的要求,则生成的构造函数为 constexpr 。 (C++11 起)

若存在某个用户定义的构造函数,则用户仍可以关键词 default 强制编译器自动生成本可隐式声明的默认构造函数。

(C++11 起)

被删除的隐式声明的默认构造函数

若下列任何一者为真,则类 T 式声明的或设为默认的默认构造函数不被定义 (C++11 前)定义为被删除 (C++11 起):

  • T 拥有无默认初始化器的 (C++11 起)引用类型成员。
  • T 拥有无用户定义默认构造函数或默认成员初始化器 (C++11 起)的 const 成员。
  • T 拥有默认构造函数被删除,或其默认构造函数有歧义或不可从此构造函数访问(无默认成员初始化器) (C++11 起)的成员。
  • T 拥有默认构造函数被删除,或有歧义,或不可从此构造函数访问的直接基类或虚基类。
  • T 拥有析构函数被删除,或不可从此构造函数访问的直接基类或虚基类。
  • T 是至少一个变体成员有非平凡默认构造函数,且变体成员无一拥有默认成员初始化器的 union 。
  • T 是拥有有非平凡默认构造函数的变体成员 M ,且含 M 的匿名联合体的变体成员无一拥有默认成员初始化器的非联合类类型。
(C++11 起)
  • T 为 union 且其所有变体成员均为 const 。

若不存在用户定义的构造函数,且隐式声明的默认构造函数非平凡,则用户仍可以关键词 delete 禁止编译器自动生成隐式定义的默认构造函数。

(C++11 起)

平凡默认构造函数

若下列全部为真则类 T 的默认构造函数为平凡(即不进行动作):

  • 构造函数非用户提供(即是隐式定义或为默认化)
  • T 无虚成员函数
  • T 无虚基类
  • T 无拥有默认初始化器的非静态数据成员。
(C++11 起)
  • 每个 T 的直接基类都拥有平凡默认构造函数
  • 每个类类型的非静态成员都拥有平凡默认构造函数

平凡默认构造函数是不进行动作的构造函数。所有与 C 语言兼容的数据类型( POD 类型)都是可平凡默认构造的。然而不同于 C 中,拥有平凡默认构造函数的对象不能通过简单地转译对齐的存储,例如以 std::malloc 分配的存储创建:要求布置 new 以形式上引入新对象并避免潜在的未定义行为。

缺陷报告

下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。

DR应用于出版时的行为正确行为
CWG 2084C++11默认成员初始化器不影响是否删除联合体的默认化默认构造函数它们阻止定义默认化默认构造函数为被删除

示例

struct A
{
    int x;
    A(int x = 1): x(x) {} // 用户定义默认构造函数
};
 
struct B: A
{
    // 隐式定义 B::B() ,调用 A::A()
};
 
struct C
{
    A a;
    // 隐式定义 C::C() ,调用 A::A()
};
 
struct D: A
{
    D(int y): A(y) {}
    // 不声明 D::D() ,因为存在另一构造函数
};
 
struct E: A
{
    E(int y): A(y) {}
    E() = default; // 显式为默认,调用 A::A()
};
 
struct F
{
    int& ref; // 引用成员
    const int c; // const 成员
    // F::F() 被隐式定义为被删除
};
 
int main()
{
    A a;
    B b;
    C c;
//  D d; // 编译错误
    E e;
//  F f; // 编译错误
}

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值