Effective C++笔记:构造/析构/赋值运算
条款05:了解C++默默编写并调用哪些函数
默认构造函数、拷贝构造函数、拷贝赋值函数、析构函数构成了一个类的脊梁,只有良好的处理这些函数的定义才能保证类的设计良好性。
当我们没有人为的定义上面的几个函数时,编译器会给我们构造默认的。
当成员变量里有const对象或引用类型时,编译器会不能合成默认的拷贝赋值函数;当一个基类把它的拷贝赋值函数定义为private时,它的派生类也不无生成默认的拷贝赋值函数,因为它无法完成基类成份的赋值。
条款06:若不想使用编译器自动生成的函数,就该明确拒绝
将拷贝构造函数和拷贝赋值操作符声明为private可以阻止类的拷贝和赋值行为,但是这样做的话,类的其他成员或友元依然可以访问,所以最终的解决方案是只声明不定义。
C++11中已经摒弃了上述的做法,而是通过删除函数来设定,只需要在对应的函数列表后加=delete即可。
我们可以为了阻止拷贝而设计一个基类,这个基类非常简单:
class Uncopyable
{
protected:
Uncopyable();
~Uncopyable();
private:
Uncopyable(const Uncopyable&);
Uncopyable& operator=(const Uncopyable&);
};
为了阻止我们设计的其他类的拷贝动作,只需要继承上面这个类即可。
条款07:为多态基类声明virtual析构函数
当你需要手动的去delete一个指向基类的指针的时候,需要将该基类的析构函数设置为virtual,这样可以让delete时,动态的删除可能的派生类的成员。
不要无端的给一个没有virtual成员函数的类的析构函数声明为virtual,因为如果要实现virtual函数必须让类携带更多的信息(存储空间)。
标准的string不含任何virtual函数,所以不要把string定义为某个定义类的基类。类似的情况也发生成STL的其他容器如vector,list,set等上。
为一个抽像基类(不能定义对象实体)声明一个纯虚函数,而且要为这个纯虚函数要提供一份定义。
class AWOV{
public:
virtual ~AWOV() = 0; //声明纯虚析构函数
};
AWOV::~AWOV(){} // pure virtual析构函数的定义
并不是多所有基类设计都是为了多态(经base class接口处理derived class对象)用途,比如说在刚开始定义的Uncopyable类。
条款08:别让异常逃离析构函数
不要让析构函数吐出异常。如果一个被析构函数调用的函数可能抛出获异常,析构函数应该捕捉任何获异常,然后吞下它们(不传播)或结束程序。
如果客户需要对某个操作函数运行期间抛出的异常做出反应,那么class应该提供一个普通函数(而非析构函数中)执行该操作。
条款09:绝不在构造和析构过程中调用virtual函数
假设你有个class继承体系,用来模拟股市交易如买进、卖出的订单等等。这样的交易一定要经过审计,所以每当创建一个交易对象,在审计日志中也需要创建一笔适当记录。
// 交易的基类
class Transaction
{
public:
Transaction();
virtual void logTransaction()const = 0;
//...
};
Transaction::Transaction()
{
// ...
logTransaction(); // 创建一笔交易的最后一步是日志里记录这笔交易
}
// 买入类
class BuyTransaction :public Transaction
{
public:
virtual void logTransaction()const; // 日志记录此型交易
//...
};
// 卖出类
class SellTransaction :public Transaction
{
public:
virtual void logTransaction()const; // 日志记录此型交易
};
现在如果我们有一笔买入的交易,我们定义了一个买入类的对象BuyTransaction b;
这个时候首先BuyTransaction类的构造函数被调用,但是首先Transaction构造函数一定会更早被调用,因为派生类的构造第一步是构造基类分份,当Transaction构造函数执行到调用logTransaction函数时,执行的是基类的版本,而并不是调用派生类里的logTransaction,虽然我们现在正在创建的是BuyTransaction。
在base class构造函数执行时,derived class的成员变量尚未初始化,如果此时真的去访问派生类的virtual函数,那这个函数还没有被初始化呢。
其实更根本的的原因是:在derived class对象的base class构造期间,对象的类型是base class而不是derived class。
类似的情况也发生在析构函数内,当BuyTransaction的析构函数执行基类的析构函数时,这时BuyTranscation类的成员将会被编译器忽略,base class执行的virtual函数都是基类里的版本。
这里如果我们还是想实现在创建日志的时候能够动态的创建不同类型的日志,我们可能通过向logTransaction中传入不同的参数来创建不同的日志,此时logTransaction将不再是一个viratual 函数。
条款10:令operator=返回一个reference* this
x=y=z=15实际上是采用右赋值的,这个连锁赋值可以解析为:x=(y=(z=15))。这里15先被赋值给z,然后其结果(更新后的z)再被赋值给y,然后其结果(更新后的y)再被赋值给x。
所以如果我们想保证y和z在完成赋值后能够对自己更新,则需要operator=返回一个本身的引用。
条款11:在operator=中处理"自我赋值"
我们定义了一个Widget的类,这个类中保存一个从heap中分配得到的指针
class Bitmap;
class Widget{
public:
Widget& operator=(const Widget& rhs);
private:
Bitmap* pb;
};
Widget& Widget::operator=(const Widget& rhs)
{
delete pb; // 释放当前类里的指针
pb = new Bitmap(*rhs.pb); // 拷贝一个新指针
return *this;
}
上面的代码看起来合理,实际上一旦出现自我赋值的时候就很不安全,因为它首先释放掉了内存空间,再从本身内存地址拷贝对象的时候,该地址所指向的对象已经不存在了。
我们可以像下面这样写出异常安全的代码
Widget& Widget::operator=(const Widget& rhs)
{
Bitmap* pOrig = pb; // 保存原来的pb
//令pb指向*pb的一个副本,即使new Bitmap异常也没问题
pb = new Bitmap(*rhs.pb);
delete pOrig;
return *this;
}
条款12:复制对象时勿忘其每一个成分
当你编写一个拷贝构造函数或拷贝赋值运算符时,请确保有的局部成员变量都被拷贝以及调用所有的基类的适当的拷贝函数。
虽然拷贝赋值操作符与拷贝构造函数做了类似的事情,但是你不能在拷贝赋值操作里调用拷贝构造函数,因为你试图对一个已经定义的对象从重定义并初始化。
如果你发现你的copy构造函数和copy assignment操作符有相近的代码,消除重复代码的做法是,建立一个新的成员函数来给两者调用。
Effective C++笔记:构造/析构/赋值运算的更多相关文章
- 【Effective C++】构造/析构/赋值运算
条款05:了解C++默默编写并调用哪些函数 默认构造函数.拷贝构造函数.拷贝赋值函数.析构函数构成了一个类的脊梁,只有良好的处理这些函数的定义才能保证类的设计良好性. 当我们没有人为的定义上面的几个函 ...
- Effective C++ 2.构造 析构 赋值运算
//条款07:为多态基类声明virtual析构函数 // 1.若基类的析构函数不定义为虚函数,由于基类的指针或引用可以指向派生类的对象,则在删除基类对象的时候可能会出错,导致破坏数据结构. // 2. ...
- 《Effective C++》第2章 构造/析构/赋值运算(2)-读书笔记
章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...
- 《Effective C++》第2章 构造/析构/赋值运算(1)-读书笔记
章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...
- EffectiveC++ 第2章 构造/析构/赋值运算
我根据自己的理解,对原文的精华部分进行了提炼,并在一些难以理解的地方加上了自己的"可能比较准确"的「翻译」. Chapter 2 构造 / 析构 / 赋值 条款 05:了解C++ ...
- Effective C++ —— 构造/析构/赋值运算(二)
条款05 : 了解C++默默编写并调用哪些函数 编译器可以暗自为class创建default构造函数.copy构造函数.copy assignment操作符,以及析构函数. 1. default构造函 ...
- Effective C++ 笔记二 构造/析构/赋值运算
条款05:了解C++默默编写并调用哪些函数 编译器默认声明一个default构造函数.一个copy构造函数.一个copy assignment操作符和一个析构函数.这些函数都是public且inlin ...
- Effective C++笔记(二):构造/析构/赋值运算
参考:http://www.cnblogs.com/ronny/p/3740926.html 条款05:了解C++默默编写并调用哪些函数 如果自定义一个空类的话,会自动生成默认构造函数.拷贝构造函数. ...
- Effective C++ -- 构造析构赋值运算
05.了解C++默默编写并调用哪些函数 编译产生的析构函数时non-virtual,除非这个类的基类析构函数为virtual 成员变量中有引用和const成员时,无法自己主动生成copy assign ...
随机推荐
- win7打开网页老是提示下载网页解决办法
方法:我的系统是windows 7 旗舰版, 解决方法可以自己手动去修复,方法是进入命令窗口. 开始--运行--cmd--sfc.exe--sfc/scannow 修复一下!
- DetachedCriteria详细使用
一.基本使用 1. 说明 Restrictions 是产生查询条件的工具类. 2. 定义 可以直接用class 创建 DetachedCriteria searDc = DetachedCriteri ...
- HDOJ 1848 Fibonacci again and again
Fibonacci again and again Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Jav ...
- PHP访问,增删改查,小结
PHP访问数据,增,删,改,查 增: 1,add.php 显示页面,利用 <form> 表单添加数据,数据添加到 name 中. 2,addchuli.php 处理页面,定义变量接受 $_ ...
- 解决 MySQL Cluster 通过 某一个MySqld节点新建表时,其他 MySqld节点 看不到表内容的问题
问题: 总共有 4 个MySqld节点,通过其中的一个节点新建表时,发现其他 MySqld节点 查不到表内容的问题,即表没有同步过来. 解决方案: 主要是因为新建表时,所选的 表引擎 错误导致的,只能 ...
- Android四大组件之Service
Android四大组件之Service Android支持服务的概念,服务是在后台运行的组件,没有用户界面,Android服务可用有与活动独立的生命周期.Android支持两种类型的服务: 本地服务: ...
- JSP基本面试的试题
JSP基本面试的试题 1.jsp有哪些内置对象作用分别是什么 答:JSP共有以下9种基本内置组件(可与ASP的6种内部组件相对应): request 用户端请求,此请求会包含来自GET/PO ...
- 解决港版A1530 ios8 联通4G 电话打不进无法接通的问题,联通4G开关开启方法
GF的iPhone5s港版A1530联通4G老是出现无法接通的问题, 根本原因是没有4G开关.港行iPhone却只有3G开关.也就是说,当启动3G时,却搜到4G信号,但是关闭时却只能关闭3G和2G.让 ...
- MySQL5.5 RPM安装的默认安装路径
MySQL5.5 RPM安装的默认安装路径 2011-06-20 10:34:32| 分类: MySQL|举报|字号 订阅 下载LOFTER客户端 由于一客户要求安装mysql- 5.5 ...
- ssh tar 命令把远程文件拉回来或推过去
ssh tar 命令把远程文件拉回来或推过去 2010-09-11 21:55:35 分类: LINUX 登录22后tar 压缩/var/log目录输出到标准输入通过管道传到本地22_log. ...