通过一个实例稍微深入理解C++复制控制过程,参考资料《C++ primer》,介绍点基本知识:

1、在C++中类通过特殊的成员函数:复制构造函数、赋值操作符和析构函数来控制复制、赋值和撤销该类的对象时会发生什么。

2、复制构造函数(copy constructor)是一种特殊的构造函数,具有单个形参,该形参(常用const)是对该类类型的引用;

  • 当定义一个新对象并用一个同类型的对象对它进行初始化时,将显示使用复制构造函数;
  • 当将该类的对象传递给函数或从函数返回该类型的对象时,将隐式使用复制构造函数。

3、复制构造函数可用于:

  • 根据另一个同类型的对象显示或隐式初始化一个对象;
  • 复制一个对象,将它作为实参传给一个函数;
  • 从函数返回时复制一个对象;
  • 初始化顺序容器中的元素;
  • 根据元素初始化式列表初始化数组元素。

4、如果我们在类体中没有定义复制构造函数,编译器就会为我们合成一个复制构造函数。与合成的默认构造函数不同,即使我们定义了其他构造函数,也会合成复制构造函数。

合成复制构造函数的行为:执行逐个成员初始化,将新对象初始化为原对象的副本。

 有些类经常有的数据成员是指针,这时我们必须对复制对象时发生的事情加以控制,否则当复制的对象析构或者删除时,对象中的指针不再指向有效对象,则会出现悬垂指针情况,这时我们必须要定义自己的复制构造函数。

5、有些类需要完全禁止复制,例如,iostream类就不允许复制,如果不想复制,则应明确禁止,可以将复制构造函数声明为private,并且不定义。

6、赋值操作符(assignment operator):同复制构造函数一样,如果在类体中没有定义赋值操作符,编译器也会为我们合成一个赋值操作符。它会执行逐个成员复制:右操作数对象的每个成员赋值给左操作数对象的对应成员。

    如:

Sales_item& Sales_item::operator=(const Sales_item &rhs)
{
isbn = rhs.isbn;
units_sold = rhs.units_sold;
revenue = rhs.revenue;
return *this;
}

7、析构函数(destructor):是构造函数的互补,当对象超出作用域或动态分配的对象(new、malloc)被删除时,将自动应用析构函数。

下面通过一个实例详细说明复制构造函数、赋值操作符、析构函数的调用机制,代码中已有详细解释,故在此不再赘述~

 1 /************************************************
2 *
3 * 内容摘要:定义Expm1类,该类给出复制控制成员和其他构造函数
4 * 用不同方式使用Expm1类型的对象:
5 * 作为非引用形参和引用形参传递,动态分配
6 * 作为函数返回值,进行赋值操作,作为元素放在vector容器中。
7 * 研究何时执行哪个构造函数和复制控制成员
8 *
9 * 日 期:2012.8.27 by Jacky Liu
10 *
11 ************************************************/
12
13 #include <vector>
14 #include <iostream>
15
16 struct Expm1
17 {
18 //默认构造函数
19 Expm1()
20 {
21 std::cout<<"Expm1()"<<std::endl;
22 }
23
24 //复制构造函数
25 Expm1(const Expm1 &)
26 {
27 std::cout<<"Expm1 (const Expm1&)"<<std::endl;
28 }
29
30 //复制操作符
31 Expm1 & operator = (const Expm1 &)
32 {
33 std::cout<<"operator = (const Expm1&)"<<std::endl;
34 return *this;
35 }
36
37 //析构函数
38 ~Expm1()
39 {
40 std::cout<<"~Expm1()"<<std::endl;
41 }
42 };
43
44 void func1(Expm1 obj) //形参为Expm1对象
45 {}
46
47 void func2(Expm1 & obj) //形参为Expm1对象的引用
48 {}
49
50 Expm1 func3()
51 {
52 Expm1 obj;
53 return obj; //返回Expm1对象
54 }
55
56 int main()
57 {
58 std::cout<<"1-------------------------------------"<<std::endl<<std::endl;
59
60 Expm1 eobj; //调用默认构造函数创建Expm1对象eobj
61
62 std::cout<<"2-------------------------------------"<<std::endl<<std::endl;
63
64 func1(eobj); //调用复制构造函数
65 //将形参Expm1对象创建为实参Expm1对象的副本
66 //函数执行完毕后调用析构函数撤销形参Expm1对象
67
68 std::cout<<"3-------------------------------------"<<std::endl<<std::endl;
69
70 func2(eobj); //形参为Expm1对象的引用,无需调用复制构造函数传递实参
71
72 std::cout<<"4-------------------------------------"<<std::endl<<std::endl;
73
74 eobj = func3(); //调用默认构造函数创建局部Expm1对象
75 //函数返回时调用复制构造函数创建作为返回值副本的Expm1对象
76 //然后调用析构函数撤销局部Expm1对象
77 //然后调用赋值操作符
78 //执行完赋值操作后
79 //调用析构函数撤销作为返回值副本的Expm1对象
80
81 std::cout<<"5-------------------------------------"<<std::endl<<std::endl;
82
83 Expm1 *p = new Expm1; //调用默认构造函数动态创建Expm1对象
84
85 std::vector<Expm1> evec(3); //调用默认构造函数创建一个临时值Expm1对象
86 //调用复制构造函数,将临时值Expm1对象复制到vector容器evec的每个元素
87 //调用析构函数撤销临时值Expm1对象
88 //按以上重复三次
89
90 std::cout<<"6-------------------------------------"<<std::endl<<std::endl;
91 delete p; //调用析构函数撤销动态创建的Expm1对象
92
93 system("pause");
94 return 0; //eobj及evec生命期结束,自动调用析构函数撤销
95
96 }

运行结果:仔细观察运行结果的执行情况,可以深入理解C++中的复制控制机制~

稍微深入点理解C++复制控制【转】的更多相关文章

  1. 深入理解redis复制原理

    原文:深入理解redis复制原理 1.复制过程 2.数据间的同步 3.全量复制 4.部分复制 5.心跳 6.异步复制 1.复制过程 从节点执行 slaveof 命令. 从节点只是保存了 slaveof ...

  2. C++继承与构造函数、复制控制

    每个派生类对象由派生类中定义的(非static)成员加上一个或多个基类子对象构成,因此,当构造.复制.赋值和撤销派生类型对象时,也会构造.复制.赋值和撤销这些基类子对象. 构造函数和复制控制成员不能继 ...

  3. C++ 复制控制之复制构造函数

    7月26日更新: 过了这么长的时间回过头来看,发现文章中有几个点说错(用红字标出): 构造函数不是只有唯一一个参数,它也可以是多参数形式,其第二参数及后继以一个默认值供应. 不是没有声明复制控制函数时 ...

  4. C++ Primer 学习笔记_67_面向对象编程 --转换与继承、复制控制与继承

    面向对象编程 --转换与继承.复制控制与继承 I.转换与继承 引言: 由于每一个派生类对象都包括一个基类部分,因此能够像使用基类对象一样在派生类对象上执行操作. 对于指针/引用,能够将派生类对象的指针 ...

  5. C++ Primer 随笔 Chapter 13 复制控制

    1.复制控制包含的内容:复制构造函数.赋值操作符.析构函数 2.复制构造函数: a. 定义:只有单个形参,而且该形参是对本类类型的引用,这样的构造函数被成为复制构造函数 b. 适用情况: (1)根据一 ...

  6. C++Primer笔记之复制控制

    复制控制这一节需要注意的地方不多,主要有以下几点: 1.定义自己的复制构造函数 什么时候需要定义自己的复制构造函数,而不用系统提供的,主要遵循以下的经验说明: 某些类必须对复制对象时发生的事情加以控制 ...

  7. C++拾遗(六)——复制控制

    年前忙了几天,到现在才算是有空休息下来.先祝大家新年快乐,心想事成:)我也会发笑脸o.o 这篇博文主要介绍定义一个类型的对象时的复制控制方式,这部分内容之前有一定的了解但又浅尝辄止,始终感觉没能找到要 ...

  8. 嗯 第二道线段树题目 对左右节点和下标有了更深的理解 hdu1556

    Color the ball Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  9. 通过laravel理解IoC(控制反转)容器和DI(依赖注入)

    原文地址: http://www.insp.top/learn-laravel-container ,转载务必保留来源,谢谢了! 容器,字面上理解就是装东西的东西.常见的变量.对象属性等都可以算是容器 ...

随机推荐

  1. (转)CentOS 7系统详细开机启动流程和关机流程

    CentOS 7系统详细开机启动流程和关机流程 原文:http://blog.csdn.net/yuesichiu/article/details/51350654 名称 bootup - 系统启动流 ...

  2. (转)Linux命令学习总结:dos2unix - unix2dos

    Linux命令学习总结:dos2unix - unix2dos 命令简介: 原文:http://www.cnblogs.com/kerrycode/p/5077969.html dos2unix是将W ...

  3. nodejs加密解密

    nodejs是通集成在内核中的crypto模块来完成加密解密. 常用加密解密模块化代码: /** * Created by linli on 2015/8/25. */ var crypto = re ...

  4. PHP中文手册1

    1.入门 关于换行 PHP 会在输出时自动删除其结束符 ?>后的一个换行.该功能主要是针对在一个页面中嵌入多段 PHP 代码或者包含了无实质性输出的 PHP 文件而设计,与此同时也造成了一些疑惑 ...

  5. 用cookie实现记住密码

    jsp-4 用cookie实现记住密码 这次就有点简单了 基本是jsp-3的代码但是有些修改 public void login(HttpServletRequest req, HttpServlet ...

  6. 在 Angularjs 中 ui-sref 和 $state.go 如何传递单个多个参数和将对象作为参数

    一: 如何传递单个参数 首先,要在目标页面定义接受的参数: 传参, 接收参数, 在目标页面的controller里注入$stateParams,然后 "$stateParams.参数名&qu ...

  7. js中===、==、!=、!===的区别

    结论 ===:先比较两个变量的类型,然后再比较值 !==:上面的值取非 ==:如果变量类型相同,比较值是否相同:类型不同,先转成相同类型,再比较值 !=:上面的值取非 代码 var num = 1; ...

  8. fn先生开篇博客

    开篇博客,也不清楚要写点啥nie,以前好像在CSDN上写过一点,但是由于本人工作号一直在变,忘了CSDN上的账号密码(尴尬...不确定这个以后会不会再次丢失呀),重新在开始吧. 记得以前,一直有朋友说 ...

  9. Java并发(五):并发,迭代器和容器

    在随后的博文中我会继续分析并发包源码,在这里,得分别谈谈容器类和迭代器及其源码,虽然很突兀,但我认为这对于学习Java并发很重要; ConcurrentModificationException: J ...

  10. Google,真的要离我们而去吗?

    Google,真的要离我们而去吗? 好怀念,真正要解决问题,还得搜google!