c++ 类–构造函数的调用时机和规避浅拷贝引发的错误。


前言

小白我最近学习了c++类作为函数参数或返回值,对这两种情况下类构造函数的调用感到很迷惑,所以专门写这篇文章来理一理。如果有错误或不足之处,欢迎大家指出,我会及时改正。

一.默认构造函数调用时机

默认构造函数在定义对象没有参数时被调用。
注意:我之前一直以为默认构造函数是指类自带的无参构造函数和拷贝构造函数,看了别人的文章才知道指的是无参构造函数,包括自己编写的无参构造函数。


二.初始化构造函数调用时机

初始化构造函数被调用的情况:
1.在定义对象时传入参数。
2.给形参类型为类的函数传入其构造函数的参数,示例代码如下:

class MyInteger{
public:

    MyInteger(int val):m_a(val) {
        cout<<"调用了初始化构造函数"<<endl;
    }
    
private:    
    int m_a;
};

void test1(MyInteger m){
     cout<<"执行测试函数"<<endl;
}

int main(){
 
   int a =10;

   test1(a);

   system("pause");
   return 0;
}

运行输出结果如下:

调用了初始化构造函数
执行测试函数
请按任意键继续. . .

主函数向test1()函数传入的数据类型为“int”而非形参的类型“MyInteger”。传入参数的过程中调用了初始化构造函数,给形参m传值的过程相当于用MyInteger m(a)的方式定义m。


三.拷贝构造函数调用时机

拷贝构造函数被调用的情况:
1.定义对象时传入同类型对象;
2.给形参类型为类的函数传入示例,示例代码如下:

class MyInteger{
public:
    MyInteger(){
        m_a = 0;
    }

    MyInteger(const MyInteger &a){
        cout<<"调用了拷贝构造函数"<<endl;
    }
    
private:    
    int m_a;
};

void test1(MyInteger m){
     cout<<"执行测试函数"<<endl;
}

int main(){

   MyInteger a;
   test1(a);
   
   system("pause");
   return 0;
}

运行输出结果如下:

调用了拷贝构造函数
执行测试函数
请按任意键继续. . .

3.函数的返回值类型为类,在函数结束后,需要调用拷贝构造函数,建立一个临时的对象,将该临时对象返回。示例代码如下:

class MyInteger{
public:
    MyInteger(){
        m_a = 0;
    }

    MyInteger(const MyInteger &a){
        cout<<"调用了拷贝构造函数"<<endl;
    }
    
private:    
    int m_a;
};

MyInteger test1(MyInteger m){
     cout<<"执行测试函数"<<endl;
     return m;
}

int main(){

   MyInteger a;
   test1(a);
   system("pause");
   return 0;
}

运行输出结果如下:

调用了拷贝构造函数
执行测试函数
调用了拷贝构造函数
请按任意键继续. . .

4.注意赋值时不会调用构造函数,代码示例如下:

MyInteger a1;
a1 = a2;

四.了解构造函数调用时机后规避浅拷贝引发的错误

浅拷贝错误:当类属性在堆上开辟空间时,使用浅拷贝容易造成重复释放的错误。以指针为例,浅拷贝只复制了指针指向的地址,并未开辟新的空间,销毁对象时会将同一片空间释放两次,引发错误。 为了避免错误发生,方式一:使用引用避免拷贝构造函数的调用。示例代码如下:
class MyInteger{
public:
    MyInteger(){
        intptr = new int;
         }
    MyInteger(int val){
        intptr = new int(val);
    }

    ~MyInteger(){
        delete intptr;
    }
    
    void show(){
        cout<<*intptr<<endl;
    }
    
private:    
    int *intptr;
};

MyInteger& test1(MyInteger &h){
    cout<<"调用测试函数"<<endl;
    return h;
}

int main(){
    MyInteger a(10);
    a.show();
    
    system("pause");
    return 0;
}

输出结果:

10
请按任意键继续. . .

方式二:自定义拷贝构造函数,采用深拷贝。
代码示例如下:

class MyInteger{
public:
    MyInteger(){

        intptr = new int;
    }
    MyInteger(int val){

        intptr = new int(val);
    }
        MyInteger(const MyInteger &m){

        intptr = new int;
        *intptr = *m.intptr;
    }

    ~MyInteger(){
      
        delete intptr;
    }
    
    void show(){
        cout<<*intptr<<endl;
    }
    
private:    
    int *intptr;
};

MyInteger test1(MyInteger h){
    cout<<"调用测试函数"<<endl;
    
    return h;
}

int main(){
   
    MyInteger a(10);

    a.show();
    
    system("pause");
    return 0;
}

输出结果:

10
请按任意键继续. . .
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值