在资源管理类中提供对原始资源的访问
资源管理类必须提供原始对象的访问接口,以便直接访问资源。
智能指针std::shared_ptr就提供成员函数get()访问原始资源。
Font GetFont() { ... } // 获取Font全局函数
ReleaseFont(Font ft) { ... } // 释放Font全局函数
ChangeFontSize(Font ft) { ... }
class Font
{
}
class FontHandle
{
public:
Font Get() { return m_font; }
// operator Font() { return m_font; } // 隐式类型转换操作符
private:
Font m_font;
}
...
FontHandle handle(GetFont()); // 1
ChangeFontSize(handle.Get()); // 2
// ChangeFontSize(handle); // 3
...
第2行中调用handle.Get()返回Font对象。如果想要FontHandle对象可以直接转换成为Font对象,可以实现其隐式类型转换操作符 operator Font (),其调用方式就变成了第3行。
当然,存在隐式类型转换操作符可能会带来意向不到的结果。
是否该提供一个显示转换函数(例如get成员函数)将RAII class转换为其底部资源,或是应该提供隐式转换,答案主要取决于RAII class别设计执行的特定工作,以及它被使用的情况。最佳设计很可能是坚持条款18的忠告:”让接口容易被正确使用,不易被误用“。通常显示转换函数如get是比较受欢迎的路子。以为它将“非故意之类型转换”的可能性最小化了。然后有时候,隐式类型转换所带来的”自然用法“也会引发天枰倾斜。
请记住:
- APIs往往要求访问原始资源(raw resources), 所以每个RAII class应该提供一个“取得其所管理之资源”的办法。
- 对原始资源的访问可能经由显示转换或隐式转换,一般而言显示转换比较安全,但隐式转换对客户比较方便。