C++按值和按址传递对象的思考和优化
C++是一门面向对象(OOP)编程语言,在这门语言中也有函数,函数的参数可以是变量数值,当然也可以是对象。所以,传统地就有关于对象是按值传递还是按址传递的讨论。
在C语言中,按值传递在很多情况下可以出色地完成任务,而且也很好理解,但是在C++中,因为有了类的对象这个可能的庞然大物(指他的数据特别大),如果还用传值的方式进行的话,会很浪费内存空间。本文就具体讨论这个问题。
在C++中,将一个对象按值传递时,会默认调用一个复制构造函数做一个这个参数的副本给函数。默认的复制构造函数名字是:类名(类名&)。请看下面代码:
#include <iostream>
using namespace std; class A
{
public:
A()
{
cout << "执行构造函数" << endl;
}
A(A&)
{
cout << "执行复制构造函数" << endl;
}
~A()
{
cout << "执行析构函数" << endl;
}
}; A func(A one)
{
return one;
}
int main (void)
{
A a;
func(a); return 0;
}
这里的执行结果是:
这里我们发现执行了一次构造函数,两次复制构造函数,三次析构函数,结果很有意思,现在分析如下:
第一次只执行构造函数是因为新建了A类的对象a,因此调用构造函数来构造对象,当main函数调用func函数的时候,将对象a的值传给了func函数,这个时候就要调用复制构造函数来做一个对象a的副本,当func函数执行完毕时,要将对象one返回,这个时候还要调用复制构造函数来做一个one对象的副本,然后调用析构函数释放func函数中的对象one,当回到main函数中的时候,由于返回的one对象没有使用,所以这个时候又要调用析构函数来释放这个返回的对象,当main函数执行完毕时,a对象的也该释放了,这个时候就要调用类的析构函数来释放a对象的空间。从上面的分析中我们看出这个过程中,有很多次调用,每次调用都要进行对象的复制,既浪费了内存,有降低了程序的执行效率。该怎么优化呢?这个时候我们可以使用传址的方式来进行,将代码修改如下:
#include <iostream>
using namespace std; class A
{
public:
A()
{
cout << "执行构造函数" << endl;
}
A(A&)
{
cout << "执行复制构造函数" << endl;
}
~A()
{
cout << "执行析构函数" << endl;
}
}; A func(A *one)
{
return (*one);
}
int main (void)
{
A a;
func(&a); return 0;
}
可以看到这里将传参的方式修改成了按址的方式,他的执行结果如下:
这里减少了一次复制构造函数和析构函数的调用,可以理解效率明显提高,但是要注意的是这里还有一对复制构造函数和析构函数的调用,这是为什么呢?仔细观察可以发现,func函数返回的是(*one),这是一个对象,而不是地址,所以他要调用复制构造函数来创建一个对象的副本,然后在析构。如果我们把func函数的返回类型修改为指针类型的时候就可以减少调用了:
#include <iostream>
using namespace std; class A
{
public:
A()
{
cout << "执行构造函数" << endl;
}
A(A&)
{
cout << "执行复制构造函数" << endl;
}
~A()
{
cout << "执行析构函数" << endl;
}
}; A *func(A *one)
{
return one;
}
int main (void)
{
A a;
func(&a); return 0;
}
这样,执行的结果就是:
这样就达到了对对象作为参数传递程序的最佳优化效果了!
C++按值和按址传递对象的思考和优化的更多相关文章
- C++ 值传递、址传递、引用传递
一.值传递 int func(int p) 值传递会在栈中开辟一块空间 p,使得p和实参的a 同值. 此时你在函数func里面对p进行任何操作都不会对原值a产生任何影响.因为a 和p本就就是两个变 ...
- [No0000B5]C# 类型基础 值类型和引用类型 及其 对象判等 深入研究1
引言 本文之初的目的是讲述设计模式中的 Prototype(原型)模式,但是如果想较清楚地弄明白这个模式,需要了解对象克隆(Object Clone),Clone其实也就是对象复制.复制又分为了浅度复 ...
- [No0000B9]C# 类型基础 值类型和引用类型 及其 对象复制 浅度复制vs深度复制 深入研究2
接上[No0000B5]C# 类型基础 值类型和引用类型 及其 对象判等 深入研究1 对象复制 有的时候,创建一个对象可能会非常耗时,比如对象需要从远程数据库中获取数据来填充,又或者创建对象需要读取硬 ...
- 1. 元信息:Meta类 2. 基于对象查询的sql优化 3. 自定义:Group_Concat() 4. ajax前后台交互
一.元信息 ''' 1. 元信息 1. Model类可以通过元信息类设置索引和排序信息 2. 元信息是在Model类中定义一个Meta子类 class Meta: # 自定义表名 db_table = ...
- QDebug &operator<<出错(根据QString来找,是不得要领的,而是应该根据QString所在的对象来思考)
程序运行后,总是崩溃在这个地方:inline QDebug &operator<<(const QString & t) 我应该用什么办法找出是哪个QString出了问题呢 ...
- POJ 3046 Ant Counting(递推,和号优化)
计数类的问题,要求不重复,把每种物品单独考虑. 将和号递推可以把转移优化O(1). f[i = 第i种物品][j = 总数量为j] = 方案数 f[i][j] = sigma{f[i-1][j-k], ...
- 【javascript】详解变量,值,类型和宿主对象
前言 我眼中的<javascript高级程序设计> 和<你不知道的javascript>是这样的:如果<javascript高级程序设计>是本教科书的话, < ...
- 【转载自netfocus博客】聚合(根)、实体、值对象精炼思考总结
1.内容摘要 最近在看DDD领域驱动设计,看到实体(Entity),值对象 (Value Object),以及聚合根(Aggregate Root) 时.对他们的关系有些模糊,不清楚.于是去找了找资料 ...
- 【js jQuery】map集合 循环迭代取值---以及 map、json对象、list、array循环迭代的方法和区别
后台给前台传来一个map @ResponseBody @RequestMapping(value = "getSys") public Map<Long,String> ...
随机推荐
- myeclipse实现Servlet实例(3) 通过继承HttpServlet接口实现
(1) 在软件公司 90%都是通过该方法开发. //在HttpServlet 中,设计者对post 提交和 get提交分别处理 //回忆 <form action="提交给?&qu ...
- Webservice-Java-Xfire
最近公司最近需要将以前提供出去的接口统一用一个标准来实现,考虑到webservice这个是标 准,因此我花时间大概学习了一下webservice,也对JAVA的几个webservice框架进行了一些小 ...
- Python源码学习之初始化(三)-PyDictObject的初始化
先来看它的定义 typedef struct _dictobject PyDictObject; struct _dictobject { PyObject_HEAD Py_ssize_t ma_fi ...
- Window7下vagrant的部署
1. 下载并安装VirtualBox 下载地址:https://www.virtualbox.org/wiki/Downloads,下载最新的安装包,接下来的安装步骤就是下一步下一步了,你懂的 ...
- Qt 界面使用自己定义控件 "提升为"
1.效果图 我做了一个很easy的样例,一个能够显示颜色的QLabel,边上有个button,点击,跳出颜色选取的Dialog,然后选择一个颜色.这个QLabel会变成什么颜色. 2.ColorLab ...
- 【足迹C++primer】46、动态存储类
动态存储类 StrVec Class Design StrVec Class Definition class StrVec { public: //构造函数 StrVec():elements(nu ...
- Visual Studio 2015 开发MVC4出现错误
在Visual Studio 2015(以下简称VS2015)中开发MVC4项目时,编译报错"当前上下文中不存在ViewBag",一直无法编译,这个是否是VS2015的Bug? 本 ...
- ASP.Net MVC概念及基本
1.1概念 MVC是一种软件设计模式,即:Model(模型).View(视图).Controller(控制器) .其主要设计目标是将用户接口和逻辑层相分离,以便开发人员更好的关注逻辑层的设计和测试, ...
- SqlCommand和SqlDataAdapter有什么区别
因为DataSet是离线的,所以SqlDataAdapter这个对象是连接DataSet和数据库的桥梁,所有对DataSet的操作(填充,更新等)都要通过他 ado.net数据访问有两种方式: 1.离 ...
- 基尔霍夫矩阵题目泛做(AD第二轮)
题目1: SPOJ 2832 题目大意: 求一个矩阵行列式模一个数P后的值.p不一定是质数. 算法讨论: 因为有除法而且p不一定是质数,不一定有逆元,所以我们用辗转相除法. #include < ...