Item 15:资源管理类需要提供对原始资源的访问 Effective C++笔记
2015-08-27 10:59
274 查看
Item 15: Provide access to raw resources in resource-managing classes.
在一个完美的设计中,所有的资源访问都应通过资源管理对象来进行,资源泄漏被完美地克服。然而世界是不完美的, 很多API会直接操作资源,尤其是一些C语言的API。总之,你会时不时地发现有需要直接访问资源, 所以资源管理对象需要提供对原始资源访问。获取资源的方式有两类:隐式地获取和显式地获取。 通常来讲,显式的资源获取会更好,它最小化了无意中进行类型转换的机会。
为了让
operator)
提供
隐式转换操作符便可以完成这个工作,比如操作系统提供了
我们封装了
通过
如果提供一个隐式类型转换运算符将
一切将会变得简单:
然而问题也随之出现:
用户无意间拷贝了一份资源!该资源并未被管理起来。这将会引发意外的资源泄漏。所以隐式转换在提供便利的同时, 也引起了资源泄漏的风险。在考虑是否提供隐式转换时,需要权衡考虑资源管理类的设计意图,以及它的具体使用场景。 通常来讲,显式的资源获取会更好,它最小化了无意中进行类型转换的机会。
除非注明,本博客文章均为原创,转载请以链接形式标明本文地址: http://harttle.com/2015/08/05/effective-cpp-15.html
在一个完美的设计中,所有的资源访问都应通过资源管理对象来进行,资源泄漏被完美地克服。然而世界是不完美的, 很多API会直接操作资源,尤其是一些C语言的API。总之,你会时不时地发现有需要直接访问资源, 所以资源管理对象需要提供对原始资源访问。获取资源的方式有两类:隐式地获取和显式地获取。 通常来讲,显式的资源获取会更好,它最小化了无意中进行类型转换的机会。
显式地获取资源
shared_ptr提供了
get方法来得到资源。
shared_ptr<Investment> pInv; void daysHeld(Investment *pi); int days = daysHeld(pInv.get());
为了让
pInv表现地更像一个指针,
shared_ptr还重载了解引用运算符(dereferencing
operator)
operator->和
operator*:
class Investment{ public: bool isTaxFree() const; }; shared_ptr<Investment> pi1(createInvestment()); bool taxable1 = !(pi1->isTaxFree()); bool texable2 = !((*pi1).isTaxFree());
隐式地获取资源
提供get方法、
operator->、
operator*已经让资源访问很方便了。然而不幸的是,程序员是懒惰的,我们还是希望能够更加简便。
隐式转换操作符便可以完成这个工作,比如操作系统提供了
FontHandle来操作字体:
FontHandle getFont(); void releaseFont(FontHandle fh); void changeFontSize(FontHandle f, int newSize);
我们封装了
Font来管理资源:
class Font{ FontHandle f; public: explicit Font(FontHandle fh): f(fh){} ~Font(){ releaseFont(f); }; FontHandle get() const { return f; } };
通过
get方法来访问
FontHandle:
Font f(getFont()); int newFontSize; changeFontSize(f.get(), newFontSize);
如果提供一个隐式类型转换运算符将
Font转换为
FontHandle,那么接受
FontHandle类型作为参数的函数将会同样地接受
Font类型。
一切将会变得简单:
class Font{ operator FontHandle() const{ return f;} }; changeFontSize(f, newFontSize);
然而问题也随之出现:
FontHandle h2 = f1;
用户无意间拷贝了一份资源!该资源并未被管理起来。这将会引发意外的资源泄漏。所以隐式转换在提供便利的同时, 也引起了资源泄漏的风险。在考虑是否提供隐式转换时,需要权衡考虑资源管理类的设计意图,以及它的具体使用场景。 通常来讲,显式的资源获取会更好,它最小化了无意中进行类型转换的机会。
除非注明,本博客文章均为原创,转载请以链接形式标明本文地址: http://harttle.com/2015/08/05/effective-cpp-15.html
相关文章推荐
- C语言程序的编译以及库的构建与使用---查漏补缺笔记
- C++ 实现观察者(Observer)模式详解
- 在C语言中转换时间的基本方法介绍
- C++ vector,list,dequeue,stack 存储结构浅析
- 基于kubuntu的C/C++开发环境搭建
- 使用GDB命令行调试器调试C/C++程序
- 使用GDB命令行调试器调试C/C++程序
- 【深入理解C++】从初始化列表和构造函数谈C++的初始化机制
- C语言中读取时间日期的基本方法
- 字符串类的实现:构造函数、析构函数、复制构造函数和赋值操作符
- C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区
- 转载C++和java多态的区别
- C语言中settimeofday函数和gettimeofday函数的使用
- VC++ 将IP字符串转为 DWORD值
- C++ sort函数
- Visual C++ 64 位迁移的常见问题
- C++ Builder XE8 安卓开发之菜单键的触发代码
- Item 14:资源管理类要特别注意拷贝行为 Effective C++笔记
- Effective C++——条款5(第2章)
- C++线程同步总结