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

拷贝构造函数的定义是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. 全文检索在 MySQL

    中就是一个 FULLTEXT 类型索引.FULLTEXT 索引用于 MyISAM 表,可以在 CREATE TABLE 时或之后使用 ALTER TABLE 或 CREATE INDEX 在 CHAR ...

  2. windows10系统右键添加cmd命令

    https://blog.csdn.net/Mr_BEelzebub/article/details/78776104 首先,在桌面新建一个文本文档. Windows Registry Editor ...

  3. java基础思维导图,让java不再难懂

    java基础思维导图,让java不再难懂 原文链接  https://my.oschina.net/u/3080373/blog/873056 最近看了一些文章的思维导图,发现思维导图真是个强大的工具 ...

  4. python中range()、list()函数的用法

      Python  range() 函数返回的是一个可迭代对象(类型是对象),而不是列表类型, 所以打印的时候不会打印列表. 函数语法: range(stop) range(start, stop , ...

  5. javascript时间日期操作

    Js获取当前日期时间及其它操作 var myDate = new Date();myDate.getYear();        //获取当前年份(2位)myDate.getFullYear();   ...

  6. vue基础——计算属性和侦听器

    计算属性——介绍 模板内的表达式非常便利,但是设计他们的初衷是用于简单计算的.在模板中放入太多的逻辑会让模板太过沉重切难以维护.如下: <div id="example"&g ...

  7. 常用类一一字符串相关类一一String类 字符串的使用

    Java字符串就是Unicode字符序列,例如“Java”就是4个Unicode字符J,a,v,a组成的. Java没有内置的字符串类型,而是在标准Java类库中提供了一个预定义的类String,每个 ...

  8. centos7 opencc 安装

    繁体字转换:http://xh.5156edu.com/jtof.php 转换的有问题http://tool.lu/zhconvert/ git网址:https://github.com/BYVoid ...

  9. 结对项目3-bug的三种状态

    这周和小伙伴结对构造程序,来深刻理解软件测试中,bug发现的三种状态. 1:不能触发Fault 2:触发Fault,但是不能触发Error 3:触发Error,但是不能产生Failure 我们完成的代 ...

  10. ASP.NET Forms身份认证详解

    ASP.NET身份认证基础 在开始今天的内容之前,我想有二个最基础的问题首先要明确: 1. 如何判断当前请求是一个已登录用户发起的? 2. 如何获取当前登录用户的登录名? 在标准的ASP.NET身份认 ...