拷贝构造函数:拷贝构造函数就是在用一个类对象来创建另外一个类对象时被调用的构造函数,如果我们没有显示的提供拷贝构造函数,编译器会隐式的提供一个默认拷贝构造函数。

拷贝构造函数的定义是X(const X& ){}

 class Test{
int m_i;
public:
Test(int i):m_i(i){}
Test(const Test& vt):m_i(vt.m_i){}//拷贝构造函数
int getI()const {return m_i;} };
 int main(){

    Test t();
Test t2(t);//调用拷贝构造函数来初始化t2对象
Test t3=t;//等价于t3(t),也是调用拷贝构造函数
return ;
}

记住拷贝构造函数的只有一个参数,并且这个参数是类类型的const引用,参数不能是普通的值传递,必须是引用。原因有二

一:如果参数是const Test vt,那么实参在传递给形参的时候,还是会给形参创建对象,分配内存。这时把实参传递给形参,仍然是是用类对象初始化类对象,还是会调用拷贝构造函数,就会形成递归调用。

二:如果传递引用,实际上是传递的一个地址,不会创建新的对象,而拷贝时也只是拷贝的一个地址大小的空间,会增加程序的效率。

当函数的形参是类的对象,调用函数时,进行形参与实参结合时使用。这时要在内存新建立一个局部对象,并把实参拷贝到新的对象中。理所当然也调用拷贝构造函数。

 void fun(Test vt){

     cout<<vt.,getI()<<endl;

 }

这个函数的参数是类的对象,会调用拷贝构造函数的

 int main(){

     Test t();
Test t2 = t;//调用一次拷贝构造函数
Test t3(t);//调用一次拷贝构造函数
fun(t);//调用拷贝构造函数
return ;
}

fun的参数变成对象的引用时,就不会再调用拷贝构造函数了。

返回值为Const 引用 以及类对象的区别

当函数的返回值是类对象,函数执行完成,返回到调用者时也会调用拷贝构造函数。理由也是要建立一个临时对象中,从函数返回的对象来初始化产生的临时对象。

 //test.h
#ifndef TEST_H
#define TEST_H
class Test
{
int m_i;
public:
Test(int i=);
Test(const Test& vt);
~Test();
int getI()const;
};
#endif //TEST_H //test.cpp
#include "Test.h"
#include<iostream>
using std::cout;
using std::endl; Test::Test(int i) :m_i(i)
{
cout << "default constructor" << endl;
} Test::Test(const Test& vt):m_i(vt.m_i){
cout << "copy constructor" << endl;
} Test::~Test()
{
} int Test::getI()const{
return m_i;
} //demo.cpp #include<iostream>
#include"Test.h"
using std::endl;
using std::cout;
void fun1(Test& vt){ cout << "m_i=" << vt.getI() << endl;
}
void fun2(Test vt){
cout << "m_i=" << vt.getI() << endl;
} Test fun3(const Test& vt){
return vt;
}
int main(){ Test t;
fun3(t);//因为fun3函数的参数是类的引用,因此不用产生临时对象。但是返回值类型为类对象会产生临时对象,
// 所以要调用一个拷贝构造函数,把从函数返回的值,拷贝到生成的临时对象中
return ;
}

我们还可以在析构函数中添加代码,来测试临时对象时什么时候释放的。

在析构函数~Test()中添加打印代码,main函数如下所示:

 int main(){

     Test t;
fun3(t);//因为fun3函数的参数是类的引用,因此不用产生临时对象。但是返回值类型为类对象会产生临时对象,
// 所以要调用一个拷贝构造函数,把从函数返回的值,拷贝到生成的临时对象中
cout << "........." << endl;
return ;
}

我们可以看到,临时对象产生后,如果没有别的对象接受,马上就销毁。如果有别的对象接受,再复制完成后销毁。但是如果mia函数中添加一句如下的代码

 int main(){

     Test t;
Test t2=fun3(t);//因为fun3函数的参数是类的引用,因此不用产生临时对象。但是返回值类型为类对象会产生临时对象,
// 所以要调用一个拷贝构造函数,把从函数返回的值,拷贝到生成的临时对象中
cout << "........." << endl;
return ;
}

此时临时对象被Test t2接受,就没有马上释放,而是当t2的生命周期完时释放。

现在增加一个函数fun4,它的返回值类型和传递的参数类型都是类的引用

 const Test& fun4(const Test& vt){
return vt;
}

然后把main的代码修改成如下的;

 int main(){

     Test t;//创建t对象,调用默认构造函数
const Test& t1=fun4(t);//传参,和返回值都没有产生临时对象。
cout << "........." << endl;
return ;
}

运行结果

我们可以看到,当返回值为类的引用时,并不会产生临时对象。这是一个非常好的改变程序效率的方法。

所以从上面的一系列代码中,我们可以看到,传参和返回值类型会影响程序的效率,因为传参类型是值类型的话,会产生临时对象,临时对象的产生,释放,拷贝,都会产生消耗,从而导致程序效率降低。

构造函数constructor 与析构函数destructor(四)的更多相关文章

  1. 构造函数constructor 与析构函数destructor(五)

    我们知道当调用默认拷贝构造函数时,一个对象对另一个对象初始化时,这时的赋值时逐成员赋值.这就是浅拷贝,当成员变量有指针时,浅拷贝就会在析构函数那里出现问题.例如下面的例子: //test.h #ifn ...

  2. 构造函数constructor 与析构函数destructor(一)

    构造函数定义:构造函数c++中在创建对象时自动调用,用来初始化对象的特殊函数. (1)构造函数的名字必须与类的名字相同,不能有返回值,哪怕是void 也不行. (2)通常情况下构造函数应声明为公有函数 ...

  3. 构造函数constructor 与析构函数destructor(二)

    (1)转换构造函数 转换构造函数的定义:转换构造函数就是把普通的内置类型转换成类类型的构造函数,这种构造函数只有一个参数.只含有一个参数的构造函数,可以作为两种构造函数,一种是普通构造函数用于初始化对 ...

  4. 构造函数constructor 与析构函数destructor(三)

    (1)构造函数初始化列表: 1 class Test{ 2 int i; 3 public: 4 Test(int vi):i(vi){}//这里的从冒号开始,到右大括号结束,这一段是构造函数初始化列 ...

  5. GCC的__attribute__ ((constructor))和__attribute__ ((destructor))

    通过一个简单的例子介绍一下gcc的__attribute__ ((constructor))属性的作用.gcc允许为函数设置__attribute__ ((constructor))和__attrib ...

  6. javascript工厂函数(factory function)vs构造函数(constructor function)

    如果你从其他语言转到javascript语言的开发,你会发现有很多让你晕掉的术语,其中工厂函数(factory function)和构造函数(constructor function)就是其中的一个. ...

  7. 【转】c++析构函数(Destructor)

    创建对象时系统会自动调用构造函数进行初始化工作,同样,销毁对象时系统也会自动调用一个函数来进行清理工作,例如释放分配的内存.关闭打开的文件等,这个函数就是析构函数. 析构函数(Destructor)也 ...

  8. C++——构造函数 constructor

    What is constructor C++中,如果你想要创建一个object,有一个函数会自动被调用(不需要programmer显式调用 ),这个函数就是constructor; construc ...

  9. 类(class)、构造函数(constructor)、原型(prototype)

    类 Class 类的概念应该是面向对象语言的一个特色,但是JavaScript并不像Java,C++等高级语言那样拥有正式的类,而是多数通过构造器以及原型方式来仿造实现.在讨论构造器和原型方法前,我可 ...

随机推荐

  1. spring-CXF-maven

    pom.xml <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w ...

  2. C# IIS 服务器 HTTP 错误 500.21 - Internal Server Error 解决办法

    <1> 管理员身份启动控制台 <2> 输入 C:\Windows\Microsoft.NET\Framework64\v4.0.30319\aspnet_regiis.exe ...

  3. leetcode171

    public class Solution { private int ConvertToC(char c) { ; switch (c) { case 'A': case 'a': rnt = ; ...

  4. JDK1.7之后switch支持string

    转自:https://blog.csdn.net/tjcyjd/article/details/9666035 在Java7之前,switch只能支持 byte.short.char.int或者其对应 ...

  5. YII assets使用

    为什么用YII assets 1.assets的作用是方便模块化,插件化的,一般来说出于安全原因不允许通过url访问protected下面的文件 ,但是我们又希望将module单独出来,所以需要使用发 ...

  6. egret 精简游戏项目

    新建一个游戏项目,我们可以删除resource文件夹下除了default.thm.json和default.res.json文件,一旦删除,当新建皮肤exml文件时会报错 还可以删除src文件夹里除了 ...

  7. ABAP-关于COMMIT WORK 和COMMIT WORK AND WAIT

    转载:https://blog.csdn.net/champaignwolf/article/details/6925019 首先说明一点:更新是异步的,更新是由SAP中UPD1和UPD2两个进程执行 ...

  8. html file 文件批量上传 以及碰到的一些问题提

    //javascript 代码 $("#submite").click(function (evt) { var arrayTr = $("#datatables&quo ...

  9. kubeadmin 部署(centos 7)

    安装指定版本docker:# yum list docker-ce --showduplicates | sort -ryum install docker-ce-18.06.1.ce-3.el7vi ...

  10. rabbitMQ 常用命令

    启动监控管理器:rabbitmq-plugins enable rabbitmq_management 关闭监控管理器:rabbitmq-plugins disable rabbitmq_manage ...