Item 5: Know what functions C++ silently writes and calls

在C++中,编译器会自己主动生成一些你没有显式定义的函数,它们包含:构造函数、析构函数、复制构造函数、=运算符。
有时为了符合既有设计。我们不希望自己主动生成这些函数。我们能够把它们显式声明为private
此时在使用这些类的客户看来,它们就像不存在一样。

  1. class Empty{
  2. public:
  3. // 默认构造函数
  4. Empty(){}
  5. // 拷贝构造函数
  6. Empty(const Empty& rhs){}
  7. // 析构函数
  8. ~Empty(){}
  9. // 赋值运算符
  10. Empty& operator=(const Empty& rhs){}
  11. };

这些编译器自己主动生成的缺省方法是能够禁用的,把它们声明为private便能解决绝大多数问题。
很多其它的讨论能够參考:Item 6: 禁用那些不须要的缺省方法-Effective
C++笔记

调用时机

当我们没有显式地定义上述这四种函数时。编译器会自己主动帮我们定义。

这些函数它们调用的时机例如以下:

  1. 构造函数:对象定义;使用其它兼容的类型初始化对象时(可使用 explicit 来避免这样的情况)
  2. 复制构造函数:用一个对象来初始化还有一对象时。传入对象參数时。返回对象时。
  3. 析构函数:作用域结束(包含函数返回)时。delete
  4. =运算符:一个对象赋值给还有一对象

为了更清晰地说明它们的调用时机。来个样例吧:

  1. Empty e1; // 默认构造函数
  2. Empty e2(e1); // 拷贝构造函数
  3. Empty e3 = e1; // 拷贝构造函数
  4. e2 = e1; // = 运算符
  5. void func(Empty e){ // 拷贝构造函数。拷贝一份參数对象
  6. return e; // 拷贝构造函数。拷贝一份返回对象
  7. // 析构函数。拷贝得到的參数对象被析构
  8. }
  9. e2 = func(e1); // = 运算符
  10. // 析构函数。返回值被析构

引用成员

当对象包括引用成员时,拷贝和赋值行为将会变得很有趣,考虑这样一个类:

  1. class Person{
  2. public:
  3. string & name;
  4. Person(string& str): name(str){ }
  5. };
  6. string s1 = "alice", s2 = "bob";
  7. Person p1(s1), p2(s2);
  8. s1 = s2;

赋值后,p1.name会指向p2.name吗?我们知道在C++中引用本身是不可改动的。

即使p1.name指向了p2.name。那么对p1.name的赋值将会影响到p2
于是。C++拒绝编译上述代码,此时我们须要手动定义一个赋值运算符。

说来奇妙。拷贝构造函数也存在相同的问题,编译器却从不抱怨。

能够正常编译,而且两个引用指向同一对象。


除非注明,本博客文章均为原创。转载请以链接形式标明本文地址: http://harttle.com/2015/07/23/effective-cpp-5.html

Item 5:那些被C++默默地声明和调用的函数 Effective C++笔记的更多相关文章

  1. 读书笔记 effective c++ Item 9 绝不要在构造函数或者析构函数中调用虚函数

    关于构造函数的一个违反直觉的行为 我会以重复标题开始:你不应该在构造或者析构的过程中调用虚函数,因为这些调用的结果会和你想的不一样.如果你同时是一个java或者c#程序员,那么请着重注意这个条款,因为 ...

  2. Effective C++(5) 了解C++默默地编写并调用哪些函数

    预热: 一个空的类,当编译器处理过之后,就包含: 一个copy构造函数 一个重载赋值操作符 一个析构函数 一个默认构造函数 Demo: class Empty() { }; // 声明一个空的类 cl ...

  3. Effective C++ 之 Item 5:了解C++默默编写并调用哪些函数

    Effective C++ chapter 2. 构造 / 析构 / 赋值运算 (Constructors, Destructors, and Assignment Operators) Item 5 ...

  4. item 5: 比起显式的类型声明,更偏爱auto

    本文翻译自modern effective C++,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 啊,简单愉快的代码: int x; 等等,讨厌!我忘了初始化x,所 ...

  5. Effective C++ 之 Item 6 : 若不想使用编译器自动生成的函数,就该明确拒绝

    Effective C++ chapter 2. 构造 / 析构 / 赋值运算 (Constructors, Destructors, and Assignment Operators) Item 6 ...

  6. 读书笔记 effective c++ Item 36 永远不要重新定义继承而来的非虚函数

    1. 为什么不要重新定义继承而来的非虚函数——实际论证 假设我告诉你一个类D public继承类B,在类B中定义了一个public成员函数mf.Mf的参数和返回类型并不重要,所以假设它们都是void. ...

  7. C#中声明、调用和配置事件的演示源码

    下面的内容是关于C#中声明.调用和配置事件的演示的内容,应该能对大伙有些好处. using System;namespace MyCollections { using System.Collecti ...

  8. Effective STL 笔记 -- Item 6 ~ 7: Container and Object Pointer

    Effective STL 笔记 – Item 6 ~ 7: Container and Object Pointer 中间两次笔记被删掉了,简单补一下: Item 3 中提到如果将对象直接放入容器中 ...

  9. EC笔记,第二部分:5.了解C++默默编写并调用哪些函数

    5.了解C++默默编写并调用哪些函数 1.C++空类 C++会为一个空类建立以下函数 (1).默认构造函数 (2).默认拷贝构造函数 (3).析构函数 (4).赋值运算符(如果成员包含引用类型或con ...

随机推荐

  1. day3 文件操作

    文件操作是在内存中进行操作的,因为文件是存储在内存中的. open函数,该函数用于文件处理: 操作文件时,一般需要经历如下步骤: (1)打开文件: (2)操作文件 一.打开文件 文件句柄 = open ...

  2. Windows Azure 初体验

    最近看到windows azure 在做活动,只需花一块钱就可以体验一个月的windows azure. 于是,我就注册了一个账号也尝试一把云时代,传送门. 注册很简单的,成功后可以看到这个界面. 然 ...

  3. Java虚拟机四:垃圾回收算法与垃圾收集器

    在Java运行时的几个数据区域中,程序计数器,虚拟机栈,本地方法栈3个区域随着线程而生,随线程而灭,因此这几个区域的内存分配和回收具有确定性,不需要过多考虑垃圾回收问题,因为方法结束或者线程结束时,内 ...

  4. 【LOJ】#2131. 「NOI2015」寿司晚宴

    题解 怎么NOI2015D1--全是一眼秒的sb题--然后我代码全都写跪一遍= = 要是NOI2015是IOI赛制我就可以AK啦(大雾) 代码能力直线下降,NOI2018滚粗预定了啊TAT 我是不是要 ...

  5. loadrunner场景中按scenario和group执行的区别

    group:多个脚本之间按照独立设置模式跑,各个脚本可以单独设置虚拟用户.运行时间等 scenario:多个脚本之间按照相同的模式跑,将总的虚拟用户数按照一定的比例分配给各个脚本

  6. 转: CreateProcessAsUser 0xC0000005访问冲突问题

    转:http://blog.csdn.net/glc22/article/details/77227367   在使用CreateProcessAsUser时出现了 0xC0000005访问冲突问题, ...

  7. git 免密码配置

    1.cd ~/ 2.touch .git-credentials   (注意文件名前面有个  ”点”) 3.打开刚刚创建的文件,写入 https://username:password@github. ...

  8. Bzoj2038/洛谷P1494 小Z的袜子(莫队)

    题面 Bzoj 洛谷 题解 考虑莫队算法,首先对询问进行分块(分块大小为\(sqrt(n)\)),对于同一个块内的询问,按照左端点为第一关键字,右端点为第二关键字排序.我们统计这个区间内相同的颜色有多 ...

  9. 转载:tar命令批量解压方法总结

    由于linux的tar命令不支持批量解压,所以很多网友编写了好多支持批量解压的shell命令,收集了一下,供大家分享: 第一:for tar in *.tar.gz;  do tar xvf $tar ...

  10. React Native 系列(五)

    前言 本系列是基于React Native版本号0.44.3写的.任何一款 App 都有界面之间数据传递的这个步骤的,那么在RN中,组件间是怎么传值的呢?这篇文章将介绍到顺传.逆传已经通过通知传值. ...