构造函数constructor 与析构函数destructor(四)
拷贝构造函数:拷贝构造函数就是在用一个类对象来创建另外一个类对象时被调用的构造函数,如果我们没有显示的提供拷贝构造函数,编译器会隐式的提供一个默认拷贝构造函数。
拷贝构造函数的定义是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(四)的更多相关文章
- 构造函数constructor 与析构函数destructor(五)
我们知道当调用默认拷贝构造函数时,一个对象对另一个对象初始化时,这时的赋值时逐成员赋值.这就是浅拷贝,当成员变量有指针时,浅拷贝就会在析构函数那里出现问题.例如下面的例子: //test.h #ifn ...
- 构造函数constructor 与析构函数destructor(一)
构造函数定义:构造函数c++中在创建对象时自动调用,用来初始化对象的特殊函数. (1)构造函数的名字必须与类的名字相同,不能有返回值,哪怕是void 也不行. (2)通常情况下构造函数应声明为公有函数 ...
- 构造函数constructor 与析构函数destructor(二)
(1)转换构造函数 转换构造函数的定义:转换构造函数就是把普通的内置类型转换成类类型的构造函数,这种构造函数只有一个参数.只含有一个参数的构造函数,可以作为两种构造函数,一种是普通构造函数用于初始化对 ...
- 构造函数constructor 与析构函数destructor(三)
(1)构造函数初始化列表: 1 class Test{ 2 int i; 3 public: 4 Test(int vi):i(vi){}//这里的从冒号开始,到右大括号结束,这一段是构造函数初始化列 ...
- GCC的__attribute__ ((constructor))和__attribute__ ((destructor))
通过一个简单的例子介绍一下gcc的__attribute__ ((constructor))属性的作用.gcc允许为函数设置__attribute__ ((constructor))和__attrib ...
- javascript工厂函数(factory function)vs构造函数(constructor function)
如果你从其他语言转到javascript语言的开发,你会发现有很多让你晕掉的术语,其中工厂函数(factory function)和构造函数(constructor function)就是其中的一个. ...
- 【转】c++析构函数(Destructor)
创建对象时系统会自动调用构造函数进行初始化工作,同样,销毁对象时系统也会自动调用一个函数来进行清理工作,例如释放分配的内存.关闭打开的文件等,这个函数就是析构函数. 析构函数(Destructor)也 ...
- C++——构造函数 constructor
What is constructor C++中,如果你想要创建一个object,有一个函数会自动被调用(不需要programmer显式调用 ),这个函数就是constructor; construc ...
- 类(class)、构造函数(constructor)、原型(prototype)
类 Class 类的概念应该是面向对象语言的一个特色,但是JavaScript并不像Java,C++等高级语言那样拥有正式的类,而是多数通过构造器以及原型方式来仿造实现.在讨论构造器和原型方法前,我可 ...
随机推荐
- leetcode976
public class Solution { public int LargestPerimeter(int[] A) { var list = A.OrderByDescending(x => ...
- 18 subprocess模块(跟操作系统交互)
1.基本概念介绍 我们经常需要通过Python去执行一条系统命令或脚本,系统的shell命令是独立于你的python进程之外的, 每执行一条命令,就是发起一个新进程,通过python调用系统命令或脚本 ...
- AJAX是什么?
AJAX的全称是Asynchronous JavaScript and XML(异步的 JavaScript 和 XML). ajax不是新的编程语言,而是一种使用现有标准的新方法.ajax是与服务器 ...
- MySQL中实现Oracle里面 rank()over ( PARTITION BY ORDER BY) 分类分组功能
各班级学生成绩测试表 select * from TMP_A; 实现目的: 按照班级分类后按照分数倒序排序 采用MySQL变量简单实现,SQL如下: SELECT a.stu_id,a.point, ...
- shell 一次移动很多个命名相似的文件
文件夹下面有很多类似下面命名的文件 aaaaaa01bbb aaaaaa01cc aaaaaa01dd aaaaaa02bbb aaaaaa02cc 要把 aaaaaa01 的文件移走 用 mv / ...
- 逻辑斯蒂回归VS决策树VS随机森林
LR 与SVM 不同 1.logistic regression适合需要得到一个分类概率的场景,SVM则没有分类概率 2.LR其实同样可以使用kernel,但是LR没有support vector在计 ...
- R画散点图、线型图、箱型图、直方图基本知识
1.导入数据 2.散点图 plot(iris[,1]~iris[,4],xlab='Length',ylab='Width',col='red',main='Length VS Width')
- PyDev找不到的问题
[PyDev找不到的问题] 下载最新版本eclipse,下载最新版jdk,即可. 参考:http://blog.csdn.net/cnweike/article/details/27096113
- mysql开通tcp远程连接
1.登陆mysql: mysql -u root mysql 2.运行下面命令 UPDATE `mysql`.`user` SET `Host` = '%' WHERE `user`.`Host` = ...
- asp.net回发页面被刷新后悔重新执行回发事件的解决方法
做项目,进行数据修改操作后,重新加载数据,本来是没问题的.但是在这个时候刷新下页面,发现修改操作又重新执行了一次,并弹出“修改成功”的提示框. 百度了下,找到以下解决方法,解决了问题: Page.Cl ...