Effective C++:条款23:宁以non-member、non-friend替换member函数
(一)
有个class来表示网页浏览器:
class WebBrowser {
public:
void clearChache();
void clearHistory();
void removeCookies();
};
很多用户会想一整个运行全部这些动作,因此WebBrowser也提供这样一个函数:clearEverything
class WebBrowser {
public:
void clearChache();
void clearHistory();
void removeCookies();
void clearEverything();
};
当然也能够由一个non-member函数调用适当的member函数而提供出来:
void clearBrowser(WebBrowser wb) {
wb.clearChache();
wb.clearHistory();
wb.removeCookies();
}
那么哪一个比較好呢?是member函数clearEverything()还是non-member函数claseBrower呢。
?
越多东西被封装,我们改变那些东西的能力也就越大。对象内的数据。越少代码看到它(訪问它),越多的数据可被封装,也就越能自由地改变对象数据,比如改变成员变量的数量。类型等等。
所以答案当然是non-member函数claseBrower。由于它导致了较大的封装性。由于它并不添加“能过訪问class内之private成分”的函数数量。这就解释了为什么clearBrowser(一个non_member non-friend函数)比clearEverything(一个member函数)更受欢迎的原因:它导致WebBrowser class有较大的封装性。
可是这里有两件事情要注意下:
第一:这里仅仅适用于non-member non-friend函数。
friend函数对class private成员的訪问权力和member函数同样,因此两者对封装的冲击也同样。
第二:仅仅因在意封装而让函数“成为class 的non-member”,并不意味着它“不能够是还有一个class的member”。假如我们能够另clearBrowser成为某工具类(utility class)的一个static member函数。
(二)
在C++中,比較自然的做法是让clearBrowser成为一个non-member函数而且位于WebBrowser所在的同一个namespace(命名空间)内:
namespace WebBrowserStuff{
class WebBrowser{...};
void clearBrowser(WebBrowser wb);
}
像clearBrowser这种函数是个“提供便利的函数”。一个像WebBrowser这种class可能拥有大量的便利函数,某些与书签有关,某些与打印有关。某些与cookie的管理有关…,通常,大多数客户仅仅对当中某些感兴趣。
没有道理一个仅仅对书签感兴趣的客户却与一个cookie相关便利函数发生编译相依关系。
分离他们最直接的方法是将他们放入不同的头文件:
//头文件webbrowser.h这一头文件针对class WebBrowser自身以及WebBrowser核心机能。
namespace WebBrowserStuff{
void clearBrowser(WebBrowser wb);
...//WebBrowser核心机能,差点儿全部客户都须要的non-member函数。
}
//头文件“webbrowserbookmarks.h”
namespace WebBrowserStuff{
class WebBrowser{...};
...//与书签相关的便利函数
}
//头文件“webbrowsercookies.h”
namespace WebBrowserStuff{
class WebBrowser{...};
...//与cookie相关的便利函数
}
C++标准程序库正是这种组织方式。数十个头文件,每一个头文件声明std的某些机能。
假设仅仅想使用vector不用#include <memory>;假设不想使用list也不须要#include <list>.这同意客户仅仅对他们所用的那一小部分系统形成编译相依。这种分割机能并不适合class成员函数,由于class 必须总体定义,不能被分割为片段。
namespace能够跨越多个源代码文件,而class不能。
将全部便利函数放在多个头文件里但隶属同一个命名空间,意味着客户能够轻松扩展这一组便利函数。
他们所要做的就是加入很多其它non-member non-friend函数到此命名空间内。比如:假设客户想写些与影像下载相关的便利函数。仅仅要在WebBrowserStuff命名空间内建立一个头文件。内含那些函数的声明就可以。新函数就像其它旧函数一样可用且整合为一体。这是class无法提供的还有一个性质,由于class定义对客户是不能扩展的。
请记住:
尽量用non-member non-friend函数替换member函数。能够添加封装性、包裹弹性(packaging flexibility)、和机能扩充性。
Effective C++:条款23:宁以non-member、non-friend替换member函数的更多相关文章
- Effective C++ -----条款23:宁以non-member、non-friend替换member函数
宁可拿non-member non-friend函数替换member函数.这样做可以增加封装性.包裹弹性(packaging flexibility)和机能扩充性.
- effective c++ 条款23 perfer nonmember nonfreind function to member function
主要的理由还是封装.nonmember nonfreind function 不能访问类private 成员变量. 这个场景是有一个类提供了一些基本功能,比如 class WebBrowser { p ...
- Effective C++ 条款23
宁non-member.non-friend顶替member性能 本节介绍笔者为什么时间来实现某些功能.择非成员函数而且是非友元函数.这样做总结一句话,就是最大限度的实现类的封装性. 封装意味着不可见 ...
- 【23】宁以non-member、non-friend替换member函数
1.non-member方法与member方法没有本质区别,对于编译器来说,都是non-member方法,因为member方法绑定的对象,会被编译器转化为non-member方法的第一个形参.non- ...
- Effective C++ -----条款36:绝不重新定义继承而来的non-virtual函数
绝对不要重新定义继承而来的non-virtual函数.
- Effective C++ -----条款06:若不想使用编译器自动生成的函数,就该明确拒绝
为驳回编译器自动提供的功能,可将相应的成员函数声明为private并且不予实现. 使用像Uncopyable这样的base class也是一种做法(即先声明一个基类,然后私有继承它).这其实有点像使用 ...
- Effective C++ -----条款02:尽量以const, enum, inline替换 #define
class GamePlayer{private: static const int NumTurns = 5; int scores[NumTurns]; ...}; 万一你的编译器(错误地)不允许 ...
- Effective C++ 条款02:尽量以const,enum,inline替换 #define
换一种说法就是宁可以编译器替换预处理器 举例 #define ASPECT_RATIO 1.653 记号ASPECT_RATIO也许从未被编译器看见:也许在编译起开始处理源码前它就被预处理器移走了,于 ...
- Effective C++ 条款六 若不想使用编译器自动生成的函数,就该明确拒绝
class HomeForSale //防止别人拷贝方法一:将相应的成员函数声明为private并且不予实现 { public: private: HomeForSale(const HomeForS ...
随机推荐
- Android设计模式之工厂模式
定义 工厂模式是我们最常用的实例化对象模式了,是用工厂方法代替new操作的一种模式.著名的Jive论坛 ,就大量使用了工厂模式,工厂模式在Java程序系统可以说是随处可见.因为工厂模式就相当于创建实例 ...
- 使用Laravel提交POST请求出现The page has expired due to inactivity错误
任何指向 web 中 POST, PUT 或 DELETE 路由的 HTML 表单请求都应该包含一个 CSRF 令牌(CSRF token),否则,这个请求将会被拒绝.
- c面试题总结
1. char *pname=new char[10];pname="asdzxc"; cout<<pname: delete pname: 该程序运行时会崩溃,原因时 ...
- 从0开始接触html--第一天学习内容总结
第一天 总结: h1-h6 p 段落 hr br 有序 ol li 无序 ul li 定义列表 dl dt dd 块级元素:独占一行,h1-h6 p hr div 行内元素:共占一行, em和i st ...
- ZOJ-3329 One Person Game (有环期望问题)
题目大意:有3个骰子,各有k1,k2,k3个面,面值为1~ki.还有一个计数器,初始值为0,统计所有的面值和.每次同时置这三个骰子,如果第一个骰子的朝上的值为a.第二个值为b.第三个值为c,那么将计数 ...
- Get和Load的区别----hibernate
Get和Load的区别
- 装载问题(load)
装载问题(load) 问题描述: 有一批共n 个集装箱要装上艘载重量为c 的轮船,其中集装箱i 的重量为wi.找出一种最 优装载方案,将轮船尽可能装满,即在装载体积不受限制的情况下,将尽可能重的集装箱 ...
- 如何迭代pandas dataframe的行
from:https://blog.csdn.net/tanzuozhev/article/details/76713387 How to iterate over rows in a DataFra ...
- Vue--关于点击当前路由,视图无法更新的解决方案
转自:https://juejin.im/post/593121aa0ce4630057f70d35 问题的根源: 用户点击当前高亮的路由并不会刷新view,因为vue-router会拦截你的路由,它 ...
- 感知器、logistic与svm 区别与联系
https://blog.csdn.net/m0_37786651/article/details/61614865 从感知器谈起 对于典型的二分类问题,线性分类器的目的就是找一个超平面把正负两类分开 ...