More Effective C++笔记(二)

三、异常

条款9:利用destructor避免泄露资源

把资源封装在对象内,通常可以在exception出现时避免资源泄露

条款10:在构造函数内阻止资源泄露

C++仅仅能删除被完全构造的对象(fully constructed objects),只有一个对象的构造函数完全运行完毕,这个对象才被完全地构造。C++拒绝为没有完成构造操作的对象调用析构函数。

在构造函数中可以使用try catch throw捕获所有的异常。更好的解决方法是通过智能指针的方式。

如果你用对应的std::unique_ptr或者shared_ptr对象替代指针成员变量,就可以防止构造函数在存在异常时发生资源泄漏,你也不用手工在析构函数中释放资源,并且你还能像以前使用非const指针一样使用const指针,给其赋值。

条款11:禁止异常exception流出析构函数destructor之外

1.可以避免terminate函数在exception传过程的栈展开机制中被调用。

如果控制权基于exception的因素离开destructor,而此时正有另一个exception处于作用状态,C++会调用terminate函数结束程序

2.可以协助确保destructor完成其所应该完成的所有事。

如果exceptiondestructor内抛出,而且没有在当地被捕获,destructor便是执行不全,仅执行到异常抛出那一点。

条款12:了解“抛出一个exception”与传递一个参数或调用一个虚函数之间的差异

函数参数和exception的传递方式有三种:by value,by reference,by pointer

  1. 抛出一个exception,不管用什么传递方式,这个exception总是会被复制,防止exception离开其生存空间后被销毁导致传递了空的exception对象,而对于by value传递,甚至会复制两次。而参数传递则不一定会复制副本。
  2. 被抛出成为exception对象被允许的类型转换动作只有两种:继承架构中的类转换,即针对base class exception而编写的catch子句可以处理类型为derived class的exceptions。第二种是允许从一个类型化指针(typed pointer)转变成无类型指针(untypedpointer),所以带有 const void* 指针的 catch 子句能捕获任何类型的指针类型异常。
  3. catch 子句匹配顺序总是取决于它们在程序中出现的顺序,所以不要让处理派生类异常的 catch 子句位于处理基类异常的 catch 子句后面。而调用虚函数时,被选中执行的是那个与对象最吻合的函数

条款13:以by reference的方式捕捉exception

catch by reference可以避免by pointer的对象删除问题,也可以避开by value的切割问题(派生类exception 对象被捕捉并被视为基类异常者,将失去派生成分,切割了子类可能需要的虚函数功能)。

条款14:明智运用exception specifications

条款15:了解异常处理的成本

四、效率

条款16:谨记80-20法则

一个程序80%的资源用域20%的代码,80%的执行时间花费在20%的代码上。

条款17:考虑使用lazy evaluation(缓式评估)

以某种方式撰写classes,使它们延缓计算,直到那些运算结果刻不容缓被迫切需要为止。可应用于:Reference Counting(引用计数)来避免非必要的对象复制、区分 operator[] 的读和写动作来做不同的事情、Lazy Fetching(缓式取出)来避免非必要的数据库读取动作、Lazy Expression Evaluation(表达式缓评估)来避免非必要的数值计算动作。

条款18:分期摊还预期的计算成本

和上一条款相反,要求超前进度的做要求以外的更多工作。

Over-eager evaluation:如果你预期你的程序常常会用到某个计算,你可以设计一份数据结构一边能够有效率的处理需求。

例如常用的caching缓存技术。

第二种是prefetching预先取出,比如stl种vecctor动态内存的分配,当需要扩张时,每次分配2倍内存。

条款19:了解临时对象的来源

  1. reference to const参数会导致有一个临时对象被产生出来绑定至该参数上。
  2. 任何时候只要你看到函数返回一个对象,就会产生临时对象(并于稍后销毁)。

条款20:协助完成返回值优化(RVO)

有时候某些操作或者函数必然要返回对象,就要产生临时对象,这无法避免,比如operator*,如果一定得以by-value方式返回对象,可以以一种特殊写法撰写函数,让他返回constructor arguments以取代对象。

此特殊的优化行为——利用函数的return点消除一个局部临时对象。

称为Return value optimization。

条款21:利用重载技术避免隐式类型转换

如下:

注意:每个“重载操作符必须获得至少一个用户自定义类信息的自变量。所以最后一个重载操作是错的。

条款22:考虑以操作符复合形式(op=)取代其独身形式(op)

  1. 一般而言,符合操作符比其对应的独身版本效率高,因为独身版本通常必须返回一个新对象,而我们必须因此承担一个临时对象的构造和析构成本。
  2. 如果同时提供某个操作符的复合形式和独身形式,便允许你的客户在效率与便利性之间做取舍。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值