《Effective C++》第2章 构造/析构/赋值运算(1)-读书笔记
章节回顾:
《Effective C++》第1章 让自己习惯C++-读书笔记
《Effective C++》第2章 构造/析构/赋值运算(1)-读书笔记
《Effective C++》第2章 构造/析构/赋值运算(2)-读书笔记
《Effective C++》第3章 资源管理(1)-读书笔记
《Effective C++》第3章 资源管理(2)-读书笔记
《Effective C++》第4章 设计与声明(1)-读书笔记
《Effective C++》第4章 设计与声明(2)-读书笔记
《Effective C++》第8章 定制new和delete-读书笔记
条款05:了解C++默默编写并调用哪些函数
当C++处理过一个空类后,编译器就会为其声明(编译器版本的):一个拷贝构造函数、一个拷贝赋值运算符和一个析构函数。如果你没有声明任何构造函数,编译器还会声明一个默认构造函数。所有这些函数都被声明为public且inline的。
例如:class Empty{};本质上是:
class Empty {
public:
    Empty() { ... }                    // default constructor
    Empty(const Empty& rhs) { ... } // copy constructor
    ~Empty() { ... }                // destructor
    Empty& operator=(const Empty& rhs) { ... } // copy assignment operator
};
说明:
(1)只有当这些函数被调用时,才会被编译器创建出来。
(2)默认构造函数和析构函数的作用例如,调用base classes和non-static成员变量的构造函数和析构函数。
(3)编译器产生的析构函数是non-virtual的,除非这个class的base class自身声明有virtual析构函数。
下面举个例子,说明编译器拒绝为class生出operator=。
template<class T>
class NamedObject
{
public:
NamedObject(std::string& name, const T& value); private:
std::string& nameValue; // this is now a reference
const T objectValue; // this is now const
} std::string newDog("Persephone");
std::string oldDog("Satch");
NamedObject<int> p(newDog, );
NamedObject<int> s(oldDog, );
p = s;
C++并不允许“让reference改指向不同对象”,所以拒绝编译赋值那一行代码,同样道理更改变const值也是非法的。如果某个base class将拷贝赋值操作符声明为private,编译器也拒绝为其derived class生出一个拷贝赋值操作符。因为编译器为derived class生成的拷贝赋值操作符想象可以处理base class成分,这是不能做到的。
条款06:若不想使用编译器自动生成的函数,就该明确拒绝
所有编译器产生的函数都是public的,所以为了阻止拷贝构造函数和拷贝赋值运算符产生,需要自行声明。下面提供两种方法来阻止copying。
(1)将成员函数声明为private而且故意不去定义,这样可以阻止拷贝。例如:iostream库中的copy构造函数和copy assignment被声明为private。
class HomeForSale {
public:
    ...
private:
    ...
    HomeForSale(const HomeForSale&);            // declarations only
    HomeForSale& operator=(const HomeForSale&);
};
说明:当客户企图拷贝对象时,编译器会阻拦他。当成员函数或friend函数拷贝对象时,连接器会阻拦它。
(2)将连接器错误移至编译器是可能的,而且是好事,越早侦测出问题越好。只要将copy构造函数和copy assignment操作符声明为private,且存在于专门为了阻止copying动作而设计的base class内。
class Uncopyable
{
protected: // allow construction
Uncopyable() {} // and destruction of
~Uncopyable() {} // derived objects...
private:
Uncopyable(const Uncopyable&); // ...but prevent copying
Uncopyable& operator=(const Uncopyable&);
};
然后让类继承Uncopyable,这样任何人包括成员函数或friend函数尝试拷贝对象时,编译器便试着生成一个copy构造函数和一个copy assignment操作符,这些函数的编译器生成版本会尝试调用其base class的对应版本,那些调用会被编译器拒绝。
注意:Uncopyable不一定得以public继承它。
请记住:为驳回编译器自动提供的机能,可将相应的成员函数声明为private并且不予实现。使用像Uncopyable这样的base class也是一种做法。
条款07:为多态基类声明virtual析构函数
C++明确指出,当derived class对象经由一个base class指针被删除,而该base class带着一个non-virtual析构函数,其结果未定义,实际执行时通常发生的是对象的derived成分没被销毁。
说明:
(1)任何class只要带有virtual函数都几乎确定应该也有一个virtual析构函数。
(2)如果class不含析构函数,通常表示它并不意图被用做一个base class。当class不企图被当作base class,令其析构函数为virtual往往是个馊主意。举例说明:
class Point // a 2D point
{
public:
Point(int xCoord, int yCoord);
~Point();
private:
int x, y;
};
如果int占32bit,那么point对象可被放入64bit缓存中。然而当point的析构函数为virtual时:
要实现出virtual函数,对象必须携带某些信息,用于在运行期决定哪一个virtual函数该被调用。这份信息通常由vptr(virtual table pointer)指针指出。vptr指向一个由函数指针构成的数组,称为vtbl(virtual table)。每一个带有virtual函数的class都有一个相应的vtbl。当对象调用某一virtual函数,实际被调用的函数取决于该对象的vptr所指的那个vtbl,编译器在其中寻找适当的函数指针。
如果Point class内含virtual函数,对象的体积会增加。两个int再加上vptr指针的大小。对象不能再被放入64bit缓存器,而且C++的Point对象也不再和其他语言(如C)内的相同声明有着一样的结构,因为其他语言的对象没有vptr,因此也就不能把它传递至其他语言写的函数。除非你明确补偿vptr,但那也丧失了可移植性。
注意:标准库string,STL容器等的析构函数均为non-virtual,所以你不能继承它们,否则可能会出现未定义行为。
令class带一个pure virtual析构函数也是很好的。假设你需要个pure class,但手头没有pure virtual函数。由于抽象class总是企图被当作base class,而又由于base class应该有个virtual析构函数。
class AWOV
{
public:
virtual ~AWOV() = ;
};
AWOV::~AWOV()
{ }
你必须为这个pure virtual析构函数提供一份定义:编译器会在AWOV的derived class的析构函数中创建一个对~AWOV()的调用动作,所以如果你不定义,连接器会报错。
请记住:
(1)并非所有的base class的设计目的都是为了多态用途。而带多态用途的base class应该声明一个virtual 析构函数。如果class带有任何virtual函数,它就应该拥有一个virtual 析构函数。
(2)class的设计目的如果不是作为base class使用,或不是为了具备多态性,就不该声明virtual析构函数。
条款08:别让异常逃离析构函数
C++并不禁止析构函数吐出异常,但它不鼓励你这样做。考虑下面一个例子:
class DBConnection
{
public:
static DBConnection create();
void close();
}; class DBConn
{
public:
~DBConn()
{
db.close();
}
private:
DBConnection db;
};
它允许客户像这样编程,而不会忘记调用close函数,关闭数据库连接。
{
    DBConn dbc(DBConnection::create());
...
}
只要能成功地调用close就好了,如果调用导致一个异常,DBConn的析构函数就会传播该异常,即允许它离开析构函数。有两个方法可以避免:
(1)如果close抛出异常就结束程序。通常通过abort完成:
DBConn::~DBConn()
{
try { db.close(); }
catch (...)
{
std::abort();
}
}
如果程序遭遇一个于析构函数间发生的错误后无法继续执行,强迫结束程序是个合理选项。因为它可以阻止异常从析构函数传播出去(那会导致未定义行为),即abort可以抢先制“不明确”行为于死地。
(2)吞下因调用close而发生的异常
DBConn::~DBConn()
{
try { db.close(); }
catch (...)
{
//制作运转记录,记下对close的调用失败
}
}
尽管吞掉异常是个坏主意,有时也比草率结束程序或不明确行为带来的风险好。
这两个办法都无法对导致close抛出异常的情况作出反应。一个较佳的策略是重新设计DBConn接口,提供一个close函数,如果客户没有主动调用close函数,就由析构函数调用。
class DBConn
{
public:
~DBConn()
{
if (!closed)
{
try
{
db.close();
}
catch (...)
{
//制作运转记录,记下对close的调用失败
}
}
}
void close()
{
db.close();
closed = true;
}
private:
DBConnection db;
bool closed;
};
把调用close的责任从DBConn析构函数转移到客户手上同时DBConn析构函数内含一层双保险。如果某个操作可能在失败时抛出异常,而又存在某种需要必须处理该异常,那这个异常必须来自析构函数以外的某个函数。因为析构函数吐出异常是危险的,总会带来“过早结束程序”或“发生不明确行为”的风险。
请记住:
(1)析构函数绝对不要吐出异常。如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常,然后吞掉它们(不传播)或结束程序。
(2)如果客户需要对某个操作函数运行期间抛出的异常做出反应,那么class应该提供一个普通函数(而非在析构函数中)执行该操作。
《Effective C++》第2章 构造/析构/赋值运算(1)-读书笔记的更多相关文章
- 《Effective C++》第2章 构造/析构/赋值运算(2)-读书笔记
		
章节回顾: <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++默默编写并调用哪些函数 默认构造函数.拷贝构造函数.拷贝赋值函数.析构函数构成了一个类的脊梁,只有良好的处理这些函数的定义才能保证类的设计良好性. 当我们没有人为的定义上面的几个函 ...
 - Effective C++ 笔记二 构造/析构/赋值运算
		
条款05:了解C++默默编写并调用哪些函数 编译器默认声明一个default构造函数.一个copy构造函数.一个copy assignment操作符和一个析构函数.这些函数都是public且inlin ...
 - Effective C++ -- 构造析构赋值运算
		
05.了解C++默默编写并调用哪些函数 编译产生的析构函数时non-virtual,除非这个类的基类析构函数为virtual 成员变量中有引用和const成员时,无法自己主动生成copy assign ...
 - Effective C++笔记(二):构造/析构/赋值运算
		
参考:http://www.cnblogs.com/ronny/p/3740926.html 条款05:了解C++默默编写并调用哪些函数 如果自定义一个空类的话,会自动生成默认构造函数.拷贝构造函数. ...
 - 【Effective C++】构造/析构/赋值运算
		
条款05:了解C++默默编写并调用哪些函数 默认构造函数.拷贝构造函数.拷贝赋值函数.析构函数构成了一个类的脊梁,只有良好的处理这些函数的定义才能保证类的设计良好性. 当我们没有人为的定义上面的几个函 ...
 - Effective C++ 2.构造 析构 赋值运算
		
//条款07:为多态基类声明virtual析构函数 // 1.若基类的析构函数不定义为虚函数,由于基类的指针或引用可以指向派生类的对象,则在删除基类对象的时候可能会出错,导致破坏数据结构. // 2. ...
 
随机推荐
- Laravel  学习记录
			
1.当配置 根目录没有指向/public 时 去掉www.blog.im/public 后的文件 需要把 public 文件夹里的..htaccess 文件放到 项目根目录.server.ph ...
 - JS正则表达式从入门到入土(9)—— test方法以及它的那些坑
			
test方法 test方法介绍 RegExp.prototype.test(str) test方法用于测试字符串参数中是否存在匹配正则表达式模式的字符串 test方法的使用 let reg = /\w ...
 - git推送到github报错:error: The requested URL returned error: 403 Forbidden while accessing https://github.com
			
最近使用git命令从github克隆仓库到版本,然后进行提交到github时报错如下: [root@node1 git_test]# git push origin mastererror: The ...
 - 尝试编辑java程序
			
尝试编译java程序 之前在用软件eclipse编译过简单的hello java程序,因为软件操作简单,后来学会了用命令符来编译程序.代码如下 public class abc { ...
 - 使用ubifs作为根文件系统的openwrt如何在进行sysupgrade时保存旧的配置
			
1.openwrt的默认方案(squashfs + jffs2) sysupgrade脚本直接调用default_do_upgrade更新设备树.内核.根文件系统,那么它是如何保存旧配置的呢?请看de ...
 - Java实习二
			
链表(java实现) Link.java public class Link{ private Node first; public Link(){ this.first = null; } //判断 ...
 - poj3977 - subset - the second time - 暴力 + 二分
			
2017-08-26 11:38:42 writer:pprp 已经是第二次写这个题了,但是还是出了很多毛病 先给出AC代码: 解题思路: 之前在培训的时候只是笼统的讲了讲怎么做,进行二分对其中一边进 ...
 - tcp westwood源代码分析
			
/* * TCP Westwood+: end-to-end bandwidth estimation for TCP * * Angelo Dell'Aera: author of the firs ...
 - org.springframework.dao.DuplicateKeyException: a different object with the same identifier value was
			
在使用hibernate框架里面的:saveOrUpdate报错: 意思就是另一个对象的id(id同值)已经被session关联了. 原因分析: 在第1步中中通过titleList.get(0)获取一 ...
 - DataReader 连接数据库完整过程和代码(Sql Server)
			
数据库名叫:Bu 有个表:A 里面有一列:ID 需要引用 using System.Data.SqlClient; 代码部分如下: SqlConnection sqlCon=new SqlConnec ...