类的构造函数学习笔记:显式调用构造函数生成临时对象,拷贝构造函数

类的构造函数学习笔记:显式调用构造函数生成临时对象,拷贝构造函数

简介  在学习《C++primer 第五版》(中文版)中第七章类的时候遇到了一个有意思的习题,原题如下:

练习7.43:假定有一个名为NoDefault的类,它有一个接受int的构造函数,但是没有默认

  在学习《C++primer 第五版》(中文版)中第七章类的时候遇到了一个有意思的习题,原题如下:

练习7.43:假定有一个名为NoDefault的类,它有一个接受int的构造函数,但是没有默认的构造函数。定义类C,C有一个Nodefault类型成员,定义C的默认构造函数

  本题的答案很简单,直接利用初始化列表给成员类提供一个默认参数即可,参考答案如下:

struct Nodefault
{
private:
    int x;
public:
     Nodefault(int y){};
};

struct C
{
private:
    Nodefault no;
public:
    C() :no(1) {};
};

  但是当时想利用一个为所有参数都提供默认实参的构造函数来等价定义默认函数,于是将C的构造函数改为下式:

    C(Nodefault x= Nodefault(1)) :no(x) {};

  初看这个构造函数的时候非常难以理解,本能的会将此行代码理解为:调用构造函数Nodefault(int)将其返回值当成x的默认初始值并利用列表初始化提供成员类no的初始值。这样理解有两个问题:

首先是构造函数是一个没有返回值的特殊函数,其次没有返回值的构造函数无法给"="提供一个有效的右值。

  显然这样的理解是错误的,正确的执行流程应该是:

  1、显式调用类Nodefault的构造函数Nodeafult(int),生成一个类的临时对象。

  2、调用拷贝构造函数将临时对象拷贝给类C的成员类no(只调用一次拷贝构造函数)。

  验证代码如下:

struct Nodefault
{
private:
    int x;
public:
     Nodefault(int y)
    {
         cout << "consttuction for Nodefault" << endl;
    };
     Nodefault(const Nodefault &c)
     {
         x = c.x;
         cout << "copy construction" << endl;
     }
     ~Nodefault() { cout << "deconstruction for Nodefault" <<endl; }
};
struct C
{
private:
    Nodefault no;
public:
    C(Nodefault x= Nodefault(1)) :no(x){
        cout << "consttuction for C" << endl;
    };
    ~C() { cout << "deconstruction for C" << endl; }
};

int main()
{
    C object;
    return 0;
}

  输出结果如下:

1 consttuction for Nodefault
2 copy construction
3 consttuction for C
4 deconstruction for Nodefault
5 deconstruction for C
6 deconstruction for Nodefault

  注意:1、只调用一次拷贝构造函数,临时对象直接拷贝给成员类。

     2、两次调用类Nodefault的析构函数:第一次在C对象构造函数结束时析构临时对象Nodefault、第二在程序结束时自动析构C后析构成员类Nodefaul

www.sobd.cc
www.somanba.cn
www.jcdi.cn
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
移动构造函数拷贝构造函数是C++中用于对象复制的两种特殊构造函数。移动构造函数用于将一个对象的资源所有权从一个对象转移到另一个对象,而拷贝构造函数用于创建一个对象并将原对象的值复制给新对象。 在编译器优化的情况下,当一个函数返回一个class对象时,编译器会尝试使用移动构造函数而不是拷贝构造函数来避免不必要的资源复制。这是因为移动构造函数可以直接将原对象的资源指针指向新对象,而不需要进行资源的复制操作,从而提高了性能。 如果程序员没有显式地实现移动构造函数,编译器会默认生成一个移动构造函数。这个默认的移动构造函数会将原对象的资源指针指向新对象,并将原对象的资源指针置为空,以确保原对象不再拥有资源的所有权。 在引用\[2\]中的示例中,Mystring实现了拷贝构造函数和移动构造函数拷贝构造函数使用深拷贝的方式复制原对象的资源,而移动构造函数则直接将原对象的资源指针指向新对象。 在引用\[3\]中的示例中,函数func返回一个Mystring对象。由于编译器优化的原因,移动构造函数会被调用来将函数内部的Mystring对象的资源所有权转移到str2对象中。因此,str2对象将拥有func函数内部Mystring对象的资源,并可以正常使用。 总结起来,移动构造函数拷贝构造函数在处理对象复制时有不同的行为。移动构造函数用于将资源所有权从一个对象转移到另一个对象,而拷贝构造函数用于创建一个对象并复制原对象的值。编译器会尝试使用移动构造函数来提高性能,如果没有显式实现移动构造函数,编译器会默认生成一个移动构造函数。 #### 引用[.reference_title] - *1* *2* *3* [C++学习笔记3:拷贝构造和移动构造](https://blog.csdn.net/pdx_ll/article/details/123882008)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值