C++11 中defaulted 和 deleted 函数

defaulted 函数

背景问题

C++ 的类有四类特殊成员函数,它们分别是:默认构造函数、析构函数、拷贝构造函数以及拷贝赋值运算符。这些类的特殊成员函数负责创建、初始化、销毁,或者拷贝类的对象。如果程序员没有显式地为一个类定义某个特殊成员函数,而又需要用到该特殊成员函数时,则编译器会隐式的为这个类生成一个默认的特殊成员函数。

示例 1:

 class X
 { 
 private: 
  int a; 
 }; 

 X obj;

在示例 1 中,程序员并没有定义类 X 的默认构造函数,但是在创建类 X 的对象 x 的时候,又需要用到类 X 的默认构造函数,此时,编译器会隐式的为类 X 生成一个默认构造函数。

该自动生成的默认构造函数没有参数,包含一个空的函数体,即 X::X(){ }。虽然自动生成的默认构造函数仅有一个空函数体,但是它仍可用来成功创建类 X 的对象 obj,示例 1 也可以编译通过。

但是,如果程序员为类 X 显式的自定义了非默认构造函数,却没有定义默认构造函数的时候,示例 2 将会出现编译错误:

示例 2:

class X
{ 
 public: 
  X(int i)
  { 
    a = i; 
  }     
 private: 
  int a; 
 }; 

 X obj;  // 错误 , 默认构造函数 X::X() 不存在

示例 2 编译出错的原因在于:类 X 已经有了用户自定义的构造函数,所以编译器将不再会为它隐式的生成默认构造函数。

如果需要用到默认构造函数来创建类的对象时,程序员必须自己显式的定义默认构造函数。例如:

示例 3:

class X
{ 
 public: 
  X(){};  // 手动定义默认构造函数
  X(int i)
  { 
    a = i; 
  }     
 private: 
  int a; 
 }; 

 X obj;   // 正确,默认构造函数 X::X() 存在

从示例 3 可以看出,原本期望编译器自动生成的默认构造函数却需要程序员手动编写了,即程序员的工作量加大了。此外,手动编写的默认构造函数的代码执行效率比编译器自动生成的默认构造函数低。

类的其它几类特殊成员函数也和默认构造函数一样,当存在用户自定义的特殊成员函数时,编译器将不会隐式的自动生成默认特殊成员函数,而需要程序员手动编写,加大了程序员的工作量。类似的,手动编写的特殊成员函数的代码执行效率比编译器自动生成的特殊成员函数低。

defaulted 函数的提出

为了解决如示例 3 所示的两个问题:
1. 减轻程序员的编程工作量;
2. 获得编译器自动生成的默认特殊成员函数的高的代码执行效率。

C++11 标准引入了一个新特性:defaulted 函数。程序员只需在函数声明后加上“=default;”,就可将该函数声明为 defaulted 函数,编译器将为显式声明的 defaulted 函数自动生成函数体。例如:

示例 4:

class X
{ 
 public: 
  X()= default; 
  X(int i)
  { 
    a = i; 
  }     
 private: 
  int a; 
 }; 

 X obj;

在示例 4 中,编译器会自动生成默认构造函数 X::X(){},该函数可以比用户自己定义的默认构造函数获得更高的代码效率。

defaulted 函数的用法及示例

defaulted 函数特性仅适用于类的特殊成员函数,且该特殊成员函数没有默认参数。例如:

示例 5:

 class X
  { 
 public: 
  int f() = default;     // 错误 , 函数 f() 非类 X 的特殊成员函数
  X(int, int) = default;  // 错误 , 构造函数 X(int, int) 非 X 的特殊成员函数
  X(int = 1) = default;   // 错误 , 默认构造函数 X(int=1) 含有默认参数
 };

defaulted 函数既可以在类体里(inline)定义,也可以在类体外(out-of-line)定义。例如:

示例 6:

class X
{ 
 public:  
   X() = default; //Inline defaulted 默认构造函数
   X(const X&); 
   X& operator = (const X&); 
   ~X() = default;  //Inline defaulted 析构函数
 }; 

 X::X(const X&) = default;  //Out-of-line defaulted 拷贝构造函数
 X& X::operator = (const X&) = default;     //Out-of-line defaulted  拷贝赋值操作符

在 C++ 代码编译过程中,如果程序员没有为类 X 定义析构函数,但是在销毁类 X 对象的时候又需要调用类 X 的析构函数时,编译器会自动隐式的为该类生成一个析构函数。该自动生成的析构函数没有参数,包含一个空的函数体,即 X::~X(){ }。例如:

示例 7:

class X
{
private:
    int x;
};

class Y : public X 
{
public:
    Y()
    {
        p = new int;
        cout << "Y构造函数\n";
    }

    ~Y()
    {
        delete p;
        cout << "Y析构函数\n";
    }

private:
    int *p;
};

int main()
{
    X *p = new Y;
    delete p;
    return 0;
}

在示例 7 中,程序员没有为基类 X 定义析构函数,当在主函数内 delete 基类指针 p 的时候,需要调用基类的析构函数。于是,编译器会隐式自动的为类 X 生成一个析构函数,从而可以成功的销毁 p 指向的派生类对象中的基类子对象。

但是,这段代码存在内存泄露的问题,当利用 delete 语句删除指向派生类对象的指针 p 时,系统调用的是基类的析构函数,而非派生类 Y 类的析构函数,因此,编译器无法析构派生类的已经在堆区分配空间 p 。

因此,一般情况下我们需要将基类的析构函数定义为虚函数,当利用 delete 语句删除指向派生类对象的基类指针时,系统会调用相应的派生类的析构函数(实现多态性),从而避免内存泄露。

但是编译器隐式自动生成的析构函数都是非虚函数,这就需要由程序员手动的为基类 X 定义虚析构函数,例如:

示例 8:

class X
{
public: 
  virtual ~X(){};     // 手动定义虚析构函数
private:
    int x;
};

class Y : public X 
{
public:
    Y()
    {
        p = new int;
        cout << "Y构造函数\n";
    }

    ~Y()
    {
        delete p;
        cout << "Y析构函数\n";
    }

private:
    int *p;
};

int main()
{
    X *p = new Y;
    delete p;
    return 0;
}

在示例 8 中,由于程序员手动为基类 X 定义了虚析构函数,当利用 delete 语句删除指向派生类对象的基类指针 p 时,系统会调用相应的派生类 Y 的析构函数以及基类 X 的析构函数,从而将派生类对象完整的销毁,可以避免内存泄露。

但是,在示例 8 中,程序员需要手动的编写基类的虚构函数的定义(哪怕函数体是空的),增加了程序员的编程工作量。更值得一提的是,手动定义的析构函数的代码执行效率要低于编译器自动生成的析构函数。

为了解决上述问题,我们可以将基类的虚析构函数声明为 defaulted 函数,这样就可以显式的指定编译器为该函数自动生成函数体。例如:

示例 9:

class X
{
public: 
   virtual ~X()= defaulted; // 编译器自动生成 defaulted 函数定义体
private:
    int x;
};

class Y : public X 
{
public:
    Y()
    {
        p = new int;
        cout << "Y构造函数\n";
    }

    ~Y()
    {
        delete p;
        cout << "Y析构函数\n";
    }

private:
    int *p;
};

int main()
{
    X *p = new Y;
    delete p;
    return 0;
}

在示例 9 中,编译器会自动生成虚析构函数 virtual X::X(){},该函数比用户自己定义的虚析构函数具有更高的代码执行效率。

deleted 函数

背景问题

对于 C++ 的类,如果程序员没有为其定义特殊成员函数,那么在需要用到某个特殊成员函数的时候,编译器会隐式的自动生成一个默认的特殊成员函数,比如拷贝构造函数,或者拷贝赋值操作符。例如:

示例 10:

 class X
 { 
 public: 
      X(); 
 }; 

 int main()
 { 
      X obj1; 
      X obj2=obj1;   // 正确,调用编译器隐式生成的默认拷贝构造函数
      X obj3; 
      obj3=obj1;     // 正确,调用编译器隐式生成的默认拷贝赋值操作符
 }

在示例 10 中,程序员不需要自己手动编写拷贝构造函数以及拷贝赋值操作符,依靠编译器自动生成的默认拷贝构造函数以及拷贝赋值操作符就可以实现类对象的拷贝和赋值。

这在某些情况下是非常方便省事的,但是在某些情况下,假设我们不允许发生类对象之间的拷贝和赋值,可是又无法阻止编译器隐式自动生成默认的拷贝构造函数以及拷贝赋值操作符,那这就成为一个问题了。

deleted 函数的提出

为了能够让程序员显式的禁用某个函数,C++11 标准引入了一个新特性:deleted 函数。程序员只需在函数声明后上“=delete;”,就可将该函数禁用。

例如,我们可以将类 X 的拷贝构造函数以及拷贝赋值操作符声明为 deleted 函数,就可以禁止类 X 对象之间的拷贝和赋值。

示例 11:

class X
{           
public: 
    X(); 
    X(const X&) = delete;  // 声明拷贝构造函数为 deleted 函数
    X& operator = (const X &) = delete; // 声明拷贝赋值操作符为 deleted 函数
}; 

 int main()
 { 
      X obj1; 
      X obj2=obj1;   // 错误,拷贝构造函数被禁用
      X obj3; 
      obj3=obj1;     // 错误,拷贝赋值操作符被禁用
 }

在示例 11 中,虽然只显式的禁用了一个拷贝构造函数和一个拷贝赋值操作符,但是由于编译器检测到类 X 存在用户自定义的拷贝构造函数和拷贝赋值操作符的声明,所以不会再隐式的生成其它参数类型的拷贝构造函数或拷贝赋值操作符,也就相当于类 X 没有任何拷贝构造函数和拷贝赋值操作符,所以对象间的拷贝和赋值被完全禁止了。

deleted 函数的用法及示例

deleted 函数特性还可用于禁用类的某些转换构造函数,从而避免不期望的类型转换。

在示例 12 中,假设类 X 只支持参数为双精度浮点数 double 类型的转换构造函数,而不支持参数为整数 int 类型的转换构造函数,则可以将参数为 int 类型的转换构造函数声明为 deleted 函数。

示例 12:

class X
{
public:
    X(double);
    X(int) = delete;
};

int main()
{
    X obj1(1.2);
    X obj2(2); // 错误,参数为整数 int 类型的转换构造函数被禁用          
}

deleted 函数特性还可以用来禁用某些用户自定义的类的 new 操作符,从而避免在自由存储区创建类的对象。例如:

示例 13:

class X
{
public:
    void *operator new(size_t) = delete;
    void *operator new[](size_t) = delete;
};

int main()
{
    X *pa = new X;      // 错误,new 操作符被禁用
    X *pb = new X[10];  // 错误,new[] 操作符被禁用
}

必须在函数第一次声明的时候将其声明为 deleted 函数,否则编译器会报错。即对于类的成员函数而言,deleted 函数必须在类体里(inline)定义,而不能在类体外(out-of-line)定义。例如:

示例 14:

class X
{
public:
    X(const X&);
};

X::X(const X&) = delete;   // 错误,deleted 函数必须在函数第一次声明处声明

虽然 defaulted 函数特性规定了只有类的特殊成员函数才能被声明为 defaulted 函数,但是 deleted 函数特性并没有此限制。非类的成员函数,即普通函数也可以被声明为 deleted 函数。例如:

示例 15:

int add(int, int) = delete;

int main()
{
    int a, b;
    add(a, b); // 错误,函数 add(int, int) 被禁用
}

值得一提的是,在示例 15 中,虽然 add(int, int)函数被禁用了,但是禁用的仅是函数的定义,即该函数不能被调用。但是函数标示符 add 仍是有效的,在名字查找和函数重载解析时仍会查找到该函数标示符。如果编译器在解析重载函数时,解析结果为 deleted 函数,则会出现编译错误。例如:

示例 16:

int add(int, int) = delete;
double add(double a, double b)
{
    return a + b;
}
int main()
{
    cout << add(1, 3) << endl;    // 错误,调用了 deleted 函数 add(int, int) 
    cout << add(1.2, 1.3) << endl;
    return 0;
}

本文转自:http://www.ibm.com/

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值