当前位置: 首页 > 编程日记 > 正文

拷贝构造函数和赋值函数的一些知识

/*******************拷贝构造函数和赋值运算符重载有以下两个不同之处***************************/

1.拷贝构造函数生成新的类对象,而赋值运算符不能。

2.由于拷贝构造函数是直接构造一个新的类对象,所以在初始化这个对象之前不用检验源对象是否和新对象相同,而复制操作符需要这个操作,另外赋值运算符中如果原来对象中有内存分配,要先把内存释放掉。

下面是String类的一个实现的部分函数,可以看出二者的区别。

 1 class String{
 2 public:
 3     String(const char * str = NULL);
 4     String(const String& other);
 5     ~String();
 6     String& operator=(const String& other);
 7 private:
 8     char *m_data;
 9 };
10 
11 String::String(const char * str){
12     if (str != NULL){
13         int length = strlen(str);
14         m_data = new char[length + 1];
15         strcpy(m_data, str);
16     }
17 }
18 
19 String::~String(){
20     delete m_data;
21 }
22 
23 String::String(const String& other){
24     int length = strlen(other.m_data);
25     m_data = new char[length + 1];
26     assert(m_data != NULL);
27     strcpy(m_data, other.m_data);
28 }
29 
30 String& String::operator=(const String& other){
31     if (this == &other){                 //这里要判断是否自我赋值
32         return *this;
33     }
34     if (m_data != NULL){                 //要检验是否有内存分配
35         delete m_data;
36     }
37     int length = strlen(other.m_data);
38     m_data = new char[length + 1];
39     assert(m_data == NULL);
40     strcpy(m_data, other.m_data);
41     return *this;
42 }

/**************一种调用拷贝构造函数和赋值函数的微妙差别*******************/

#include<iostream>
#include<assert.h>
using namespace std;class B{
public:B():data(0){cout << "default constructor" << endl;}B(int i):data(i){cout << "constructed by parameter " << data << endl;}B (B &b){data = b.data;cout << "copyed by parameter " << data << endl;}B & operator=(const B& b){this->data = b.data;cout << "=     by parameter " << data << endl;return *this;}
private:int data;
};

void test(){
    B b1;
    B b2 = b1;
    B b3;
    b3 = b1;
}
int main(){

  test();
system("pause");return 0; }

test()函数和system("pause")是为了不退出main函数,可以看到析构函数执行。运行之后得到以下结果。

default constructor
copyed by parameter 0
default constructor
=     by parameter 0

注意仔细看test函数里的代码段。

b2调用的是拷贝构造函数,而b3调用的是赋值函数。这两者是不同的。

/*********************关于拷贝构造函数和赋值函数的临时对象问题******************************/

看以下代码,它的输出会是什么。

 1 #include<iostream>
 2 #include<assert.h>
 3 using namespace std;
 4 
 5 class B{
 6 public:
 7     B():data(0){
 8         cout << "default constructor" << endl;
 9     }
10     ~B(){
11         cout << "destructed by parameter " << data << endl;
12     }
13     B(int i):data(i){
14         cout << "constructed by parameter " << data << endl;
15     }
16     B (B &b){
17         data = b.data;
18         cout << "copyed by parameter " << data << endl;
19     }
20     B & operator=(const B& b){
21         this->data = b.data;
22         cout << "=     by parameter " << data << endl;
23         return *this;
24     }
25 private:
26     int data;
27 };
28 
29 B play(B b){
30     return b;
31 }
32 
33 void test(){
34     play(1);
35     B t1 = play(2);
36     B t2;
37     t2 = play(3);
38 }
39 
40 int main(){
41 
42     test();
43     system("pause");
44     return 0;
45 }


这个程序比上一个增加了一个play()函数和析构函数的输出。看到输出结果后,有一些疑惑。以下为输出结果,为方便起见,给它们编号。

(1)constructed by parameter 1                                            
(2)copyed by parameter 1
(3)destructed by parameter 1
(4)destructed by parameter 1
(5)constructed by parameter 2
(6)copyed by parameter 2
(7)destructed by parameter 2
(8)default constructor
(9)constructed by parameter 3
(10)copyed by parameter 3
(11)destructed by parameter 3
(12)=     by parameter 3
(13)destructed by parameter 3
(14)destructed by parameter 3
(15)destructed by parameter 2

如果有疑问,可以先了解下面三点。

1.用同一个类的源对象构造一个目标对象是,会调用拷贝构造函数来构造目标对象,如果没有定义拷贝构造函数,将会调用默认的拷贝函数来构造目标对象。

2.当类有一个带有一个参数的构造函数时,可以用这个参数同类型的数据初始化这个对象,默认会调用这个构造函数。

3.当一个函数的返回值为一个类的对象时,如果在调用函数中(注意是调用函数,不是被调用函数),没有定义一个对象来接收这个返回值,会用返回一个临时对象保存返回对象的值。在被调用函数(注意是被调用函数)结束时,这个临时对象被销毁。而当有一个接收对象时,就将返回对象赋值给接收对象,这个返回对象在调用函数(注意是调用函数)结束时调用析构函数。

第一点体现在程序35行,上面讲过,这是会调用拷贝构造函数而不是赋值函数。

第二点体现在34行,play(1)的类型是整型,而类B中有一个带int类型的构造函数,当实参1传给形参b,会调用这个构造函数。

第三点体现在34行和37行。play(1)函数被调用,返回一个类对象,而此时没有对象接收(在左边接受赋值),所以会返回一个临时对象,而这个临时对象在被调用函数play结束时调用析构函数销毁,输出(4)destructed by parameter 1;t2 = play(3);语句中play(3)被调用,返回一个对象,而此时有对象(t2)接收,所以调用赋值函数赋值给t2,在调用函数(test)结束时,这个对象才被销毁输出(13)destructed by parameter 3。

所以,上面的输出的含义分别是:

constructed by parameter 1                  //用1构造参数b
copyed by parameter 1                        //用b构造一个临时对象
destructed by parameter 1                   //参数b被析构
destructed by parameter 1                   //临时对象被析构
constructed by parameter 2                 //用2构造参数b
copyed by parameter 2                        //用b构造t1
destructed by parameter 2                   //参数b被析构
default constructor                             //构造t2
constructed by parameter 3                 //用3构造参数b
copyed by parameter 3                       //用b拷贝一个临时对象
destructed by parameter 3                  //参数b被析构
=     by parameter 3                           //调用赋值函数=()初始化t2
destructed by parameter 3                 //临时对象被析构
destructed by parameter 3                 //t2被析构
destructed by parameter 2                 //t1被析构

转载于:https://www.cnblogs.com/piginthetree/p/3898487.html

相关文章:

代码重构之三种取代类型码(类、子类、状态对象或策略对象)的方式辨析

1.以类取代类型码 适用情况&#xff1a;类之中有一个数值类型码&#xff0c;但它并不影响类的行为。 重构手段&#xff1a;以一个新的类替换该数值类型码。 重构类图示意&#xff1a; 这里的“不影响类的行为”是什么意思呢&#xff1f; 类型码往往和switch语句一起出现&#…

NHibernate之旅(11):探索多对多关系及其关联查询

本节内容 多对多关系引入多对多映射关系多对多关联查询1.原生SQL关联查询2.HQL关联查询3.Criteria API关联查询结语多对多关系引入 让我们再次回顾在第二篇中建立的数据模型&#xff1a; 在图上&#xff0c;我已经清晰的标注了表之间的关系&#xff0c;上两篇分析Customer和Ord…

自动化运维—saltstack

2019独角兽企业重金招聘Python工程师标准>>> 自动化运维——saltstack 、ansible 一、自动化运维介绍 传统运维&#xff1a;传统运维效率低&#xff0c;大多工作需要人工完成&#xff0c;工作繁琐&#xff0c;容易出错&#xff0c;每日重复做相同的事情&#xff0c;…

史上最浅显易懂的Git教程!

Git初学者很好的一篇教程 mark : ) http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000 转载于:https://www.cnblogs.com/anthony0859/p/3900327.html

狎昵关系和依恋情结辨诠

Inappropriate Intimacy&#xff08;狎昵关系&#xff09; 表现&#xff1a;两个classes过于亲密&#xff0c;花费太多时间去探究彼此的private成分 解决&#xff1a;你可以采用 Move Method 和 Move Field 帮它们划清界线&#xff0c;从而减少狎昵行径。你也可以看看是否运用…

win2000.win2003关闭端口详解--防黑必备

我相信有很多人都不知道自己开了什么端口.更加不知道怎么关闭端口. 你可以用查看端口的软件查看. 也可以通过在运行里输入"cmd" 在弹出的cmd命令行里输入 netstat -an 来查看自己开放端口.ip地址的后面的就是端口号. 以下是我自己写的一篇关于关闭端口的详细步骤和多…

网站基于vs,复选框,单选款

前端代码&#xff1a; <% Page Language"C#" AutoEventWireup"true" CodeFile"Default2.aspx.cs" Inherits"Default2" %><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.or…

thinphp 整合ueditor

我的ueditor是部署在public/editor 部署前台页面 <script type"text/javascript" > var UEDITOR_HOME_URL: "__PUBLIC__/ueditor/" </script><script id"container" name"$des" type"text/plain">这里写你…

笔画宽度变化(C++和matlab算法)

最近一直在看工作方面的书籍&#xff0c;把论文的事情搁置了&#xff0c;之前承诺的贴代码的事一直拖。现在把代码整理发上来&#xff0c;只有核心部分的&#xff0c;都不是我写的&#xff0c;我是网上整理下载的&#xff0c;matlab代码的效果比较差。 全部文件网盘下载地址:ht…

bzoj1227: [SDOI2009]虔诚的墓主人(树状数组,组合数)

传送门 首先&#xff0c;对于每一块墓地&#xff0c;如果上下左右各有$a,b,c,d$棵树&#xff0c;那么总的虔诚度就是$C_k^a*C_k^b*C_k^c*C_k^d$ 那么我们先把所有的点都给离散&#xff0c;然后按$x$为第一关键字&#xff0c;$y$为第二关键字&#xff0c;那么同一横坐标的一定在…

[导入]源代码版本控制(一)

开发过程当中源代码的版本控制一直是个大问题。项目规模小了还好办&#xff0c;人的脑子还能记过来&#xff0c;项目大了&#xff0c;可能用各式各样的表格来记录版本信息和源代码内容&#xff0c;但这个办法本身的文档组织又是个问题&#xff0c;谁来维护&#xff1f;谁来更改…

重构技巧分别能够解决哪些代码味道

1.提炼类可以解决的5种代码味道&#xff1a; 过大类 重复代码 基本类型偏执 令人迷惑的暂时值域 狎昵关系 2.将类内联化可以解决的3种代码味道 冗赘类 夸夸其谈的未来性 霰弹式修改 3.隐藏委托关系解决的2种代码味道 狎昵关系 过度耦合的消息链 4.复制被监视的数据 过大类 5.以…

python爬取电影和美食数据实战

本文使用的是requests正则来匹配网页内容&#xff0c;对于数据量较多的采用了多线程抓取的方法&#xff0c;共3个案例&#xff0c;分别是抓取猫眼电影TOP100榜单和淘票票正在热映的电影信息、以及美团的美食数据。这几个案例采用的方法大同小异。1、首先选择想要爬取的网站2、确…

Asp.Net页面执行流程分析

在我的上一篇文章中说到了HttpModule、HttpHandle的简单使用&#xff0c;我们可以利用它们在页面请求的过程中加入自己的事件处理程序。那么在一个aspx页面请求时后台到底做了什么&#xff1f;当然asp.net做了很多事情&#xff0c;过程也比较复杂&#xff0c;本文主要分析一下大…

正则验证非法字符

function regText(text){var reg /^[\s\u4e00-\u9fa5a-z0-9_-]{0,}$/;if(!reg.exec(text)){console.log("非法字符")}else{console.log("有效字符")} } regText("abc") 验证 &#xff1a;汉字、英文、数字、下划线、中划线、空格 转载于:https…

活动排序工具之双代号网络(AOA)与单代号网络(AON)[cont.]

箭线图ADM/双代号网络AOA 图示 箭线表示活动 节点表示一个活动的开始或结束 三要素&#xff1a;结点、箭线、线路 唯一使用虚活动的活动排序工具&#xff0c;虚活动用虚线箭头表示&#xff0c;没有历时&#xff0c;不需资源&#xff0c;只表达活动关系的需要 只使用一种活动之…

并发任务的可视化

一、任务要求&#xff1a;在linux系统中设计一个父进程&#xff0c;三个子进程(A,B,C)。子进程A,B同时被父进程启动来计算&#xff08;不实现具体的计算任务&#xff0c;先用CPU空跑来代替&#xff09;。进程A计算5分钟&#xff0c;而进程B计算8分钟。当进程A,B都计算完成后才能…

银监会警示担保圈贷款风险 联保贷款变异 防多米诺效应

互保联保本是解决小微企业以及农村金融贷款需求的重要创新&#xff0c;但却在部分行业、部分地区逐渐变异&#xff0c;成为引发风险事件的诱因。 据媒体报道&#xff0c;银监会近日发文要求加强企业担保圈贷款风险的防范和化解工作。银监会警示&#xff0c;担保圈企业风险较高的…

SharePoint 2007 系列(4) -Site Settings

Site administration 转载于:https://www.cnblogs.com/xuxiaoguang/archive/2008/11/05/1326913.html

软件项目管理重点总结

文章目录概论走进项目管理把控环境&#xff0c;控制过程整合项目资源控制项目范围保障项目进度驾驭项目成本保证项目质量协调项目人力资源改善项目的沟通应对项目风险关注项目的采购和外包概论 项目的定义&#xff1a;为创造一个特定的产品、服务或者成果而采取的临时性的努力…

jQuery发送含有数组参数的ajax请求以及后台Struts2的OGNL解析错误

当使用jquery1.3以上版本时&#xff0c;进行ajax参数传值时&#xff0c;会出现以下的一个错误: ognl.ExpressionSyntaxException: Malformed OGNL expression: f[] [ognl.ParseException: Encountered " "]" "] "" at line 1, column 3. 这个错…

数据绑定以及Container.DataItem绑定技巧

数据绑定以及Container.DataItem绑定技巧 灵活的运用数据绑定操作绑定到简单属性:<%#UserName%>绑定到集合:<asp:ListBox id"ListBox1" datasource<%# myArray%> runat"server">绑定到表达式:<%#(class1.property1.ToString() "…

LeetCode 76. Minimum Window Substring / 567. Permutation in String

76. Minimum Window Substring 典型Sliding Window的问题&#xff0c;维护一个区间&#xff0c;当区间满足要求则进行比较选择较小的字串&#xff0c;重新修改start位置。 思路虽然不难&#xff0c;但是如何判断当前区间是否包含所有t中的字符是一个难点&#xff08;t中字符有重…

计算机网络中的协议数据单元的控制信息主要包括哪些内容

在计算机网络的数据传输过程中会对数据进行封装&#xff0c;俗称加头(链路层还会加尾)&#xff0c;增加的控制信息主要包括以下内容&#xff1a; 地址(Address)&#xff1a;用来标识发送端或接收端差错检测编码(Error-detecting code)&#xff1a;用于差错检测或纠正协议控制(…

jQuery 超屏加载

jQuery 超屏加载&#xff0c;当文档超出屏幕的高度时&#xff0c;加载最新下个列数据 $(window).scroll(function () {var height $(document).height(); //页面的高度var keheight $(window).height(); //浏览器可视的高度var sheight $(document).scrollTop(); //滚动的高…

自己动手,打造轻量级VSCode/C#环境代替LinqPad

.Net 的项目都挺重的&#xff0c;一直想找一个轻量级的 CSharp 环境&#xff0c;能像Python那样&#xff0c;选一个文件就能跑的。之前用的是 LinqPad&#xff0c;但它的缺点也很明显&#xff1a; &#xff08;1&#xff09; 不付费&#xff0c;自动完成不能用&#xff08;…

让html:error只显示第一条错误信息

struts-config.xml 中的 <plug-in className"org.apache.struts.validator.ValidatorPlugIn"> 里面加上 <set-property property"stopOnFirstError" value"true"/> 就可以了 </plug-in> 转载于:https://www.cnblogs.com/kakak…

(C++)除基取余法:将十进制数转化为Q进制数

所谓基&#xff0c;就是指将要转换成的进制Q。 除基取余的意思就是&#xff1a;每次将待转换数除以Q&#xff0c;然后将得到的余数作为低位存储&#xff0c;而商则继续除以Q并重复上面的操作&#xff0c;直至商0时&#xff0c;将所有位从高到低输出就可以得到Q进制数。 代码实…

《C++ Primer 4th》读书笔记 第5章-表达式

原创文章&#xff0c;转载请注明出处&#xff1a; http://www.cnblogs.com/DayByDay/p/3912114.html 转载于:https://www.cnblogs.com/DayByDay/p/3912114.html