Effective C++:条款14:在中小企业资源管理copying表现
(一)
在一项条款说法auto_ptr和tr1::share_ptr适合heap-based资源。然而,并非所有的资源都heap-based的。换句话说不tr1::shared_ptr 和 auto_ptr 总是适合作为资源管理器。管理类型。
如果Mutex类型通过lock和unlock两组函数进行相互排斥器的锁定和解锁,可能我们希望和auto_ptr一样的行为。在某个智能类型析构时主动调用unlock进行解锁。
比方以下的代码:
void lock(Mutex* pm);
void unlock(Mutex* pm);
class Lock{
public:
explicit Lock(Mutex* pm) : mutexPtr(pm)
{
lock(mutexPtr); //获得资源
} ~Lock(){ unlock(mutexPtr); } //释放资源
private:
Mutex *mutexPtr;
};
当我们Lock对象进行copy时会发生什么,例如以下:
Mutex m;
Lock m11(&m);
Lock m12(m11); //进行copy行为
导致的恶果就是将会对同一个资源释放两次。
(二)解决方法:
解决的方法一:禁止复制。
假设复制动作对于RAII class并不合理的话。我们便应该禁止之!
回见条款6。声明一个Uncopyable类。把RAII class的copying操作声明为Uncopyable类的private。
然后再继承之:
class Lock : private Uncopy{ //禁止复制,见条款6
...
};
解决方法二:对底层资源祭出“引用计数法”(reference counting)。
有时候我们希望保有资源,直到它的最后一个使用者(某对象)被销毁。
通常仅仅要引用一个tr1::shared_ptr成员变量便可实现出reference_counting copy行为。但此时,tr1::shared_ptr的缺省行为是“当引用次数为0时删除其所指物”,那不是我们所要的行为,我们想要做的释放动作是解除锁定而非删除!
幸运的是。tr1::shared_ptr同意我们指定所谓的“删除器”(deleter),当引用次数为0时被调用,(此机能并不存在于auto_ptr)。
删除器对tr1::share_ptr构造函数而言是可有可无的第二个參数:
class Lock{
public:
explicit Lock(Mutex* pm) : mutexPtr(pm, unlock); //以某个Mutex初始化shared_ptr,并以unlock函数为删除器
{
lock(mutexPtr.get());
}
private:
tr1::shared_ptr<Mutex> mutexPtr;
};
所以在以后当引用次数为0时unlock函数就会被调用。
假设没有设置unlock这个删除器,那么对象所指资源就会被删除,那不是我们所想要的结果。
解决方法三:复制底部资源
能够对一份资源拥有其随意数量的复件。
复制资源管理对象时是“深度拷贝”。不仅指针会被制作出一个复件。并且会创建一个新的内存。
标准字符串类型是由指向heap内存的指针构成。当这样一个字符串对象被复制,不论指针还是其所指内存都会被复制一个复件。
这种字符串展现“深度复制”。
解决方法四:转移底部资源的拥有权
你希望确保仅仅有一个RAII对象指向一个未加工资源,即使RAII被复制依旧如此。
资源的拥有权从被复制转移到目标物。这事实上就是auto_ptr的复制意义。
请记住:
(1)复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为。
(2)普遍而常见的RAII class copying行为是:抑制copying、引用计数法的实现。
但其他动作也很可能被实现。
版权声明:本文博主原创文章,博客,未经同意不得转载。
Effective C++:条款14:在中小企业资源管理copying表现的更多相关文章
- Effective C++ -----条款14: 在资源管理类中小心copying行为
复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为. 普遍而常见的RAII class copying行为是:抑制copying(使用私有继承 ...
- 条款14:在资源管理类中小心copying行为
请牢记: 1.复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为. 2.普遍常见的RAII class copying行为是:抑制copyin ...
- 条款14:在资源管理类中心copying行为(Think carefully about copying behavior in resource-manage classes)
NOTE: 1.复制RAII 对象必须一并赋值它所管理的资源,所以资源的copying行为决定RAII对象的copying行为. 2.普遍而常见的RAII class copying 行为是: 抑制c ...
- Effective C++ 条款14
在资源管理器中小心copying行为 上节是对资源的管理说明.有时候我们不能依赖于shared_ptr或者auto_ptr,所以我们须要自己建立一个资源管理类来管理自己的资源. 比如建立一个类来管理M ...
- Effective C++ Item 14 Think carefully about copying behavior in resource-managing classe
In C++, the only code that guaranteed to be executed after an exception is thrown are the destructor ...
- Effective C++ -----条款15:在资源管理类中提供对原始资源的访问
APIs往往要求访问原始资源(raw resources),所以每一个RAII class应该提供一个“取得其所管理之资源”的办法. 对原始资源的访问可能经由显示转换(.get()成员函数或者指针取值 ...
- 条款14:在资源管理类中小型coping的行为
首先假设对于一个mutex互斥器对象,有lock以及unlock两个函数可用: void lock(Mutex * pm); void unlock(Mutex * pm); 那么为了防止资源忘记被释 ...
- 《Effective C++》第3章 资源管理(1)-读书笔记
章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...
- Effective C++笔记(三):资源管理
参考:http://www.cnblogs.com/ronny/p/3745098.html 资源:动态分配的内存.文件描述器.互斥锁.图形界面中的字型与笔刷.数据库连接以及网络sockets等, ...
随机推荐
- AutoFac使用方法总结:Part I
注册部分 使用RegisterType进行注册 [Fact] public void can_resolve_myclass() { var builder = new ContainerBuilde ...
- ORACLE Install (10g r2) FOR Red Hat Enterprise Linux Server release 5.5 (64 bit) (转)
OS Info----------# cat /etc/redhat-releaseRed Hat Enterprise Linux Server release 5.5 (Tikanga)# cat ...
- sql 与linq的转换
1. left Join 原始sql select t.[MINTAccountIdentifier] from BSS_Tenant t left join BL_SAPCustomer s on ...
- Byte[]和BASE64之间的转换
一. BASE64编码 把byte[]中的元素当做无符号八位整数转换成只含有64个基本字符的字符串,这些基本字符是: l 大写的A-Z l 小写的a-z l 数字0-9 l '+' 和 '/' l 空 ...
- CareerCup它1.8 串移包括问题
[称号] 原文: 1.8 Assume you have a method isSubstring which checks if one word is a substring of another ...
- List、Map和Set实现类
List.Map和Set实现类 1.List实现类 (1)ArrayList (2)Vector 2.Map实现类 (1)HashMap (2)Hashtable 3.Set实现类 (1)HashSe ...
- 【SQL】Oracle的PL/SQL语法及其拓展数据类型总结
PL/SQL语法 PL/SQL程序由三部分组成,声明部分.执行部分.异常处理部分. 模板: DECLARE /*变量声明*/ BEGIN /*程序主体*/ EXCEPTION /*异常处理部分*/ E ...
- UVA 11402 - Ahoy, Pirates!(段树)
UVA 11402 - Ahoy, Pirates! 题目链接 题意:总的来说意思就是给一个01串,然后有3种操作 1.把一个区间变成1 2.把一个区间变成0 3.把一个区间翻转(0变1,1变0) 思 ...
- C#启动进程之Process
在程序设计中,我们经常会遇到要从当前的程序跳到另一个程序的设计需求.也就是当前进程创建另一个进程.C#提供了Process使得我们很方便的实现. 1.Process基本属性和方法 Id //进程的Id ...
- therefore/so/hence/then/accordingly/Thus
这几个词的区别大致可从以下几方面去看:1.therefore adv.因此, 所以=for that reason=consequently常用于连接两个并列分句,其前加“and”或分号“:”.He ...