读书笔记_Effective_C++_条款三十八:通过复合塑模出has-a或者is-implemented-in-terms-of
如果说public是一种is-a的关系的话,那么复合就是has-a的关系。直观来说,复合就是在一个类中采用其他类的对象作为自身的成员变量,可以举个例子,像下面这样:
class Person
{
private:
string Name; // 复合string类型的变量
PhoneNumber HomeNumber; // 复合PhoneNumber对象
PhoneNumber TelephoneNumber;
};
我们一般会说人有名字,有家庭电话,有手机电话等,但我们一般不会说人是一个名字,或者人是一个家庭电话等。所以在这里,我们并不会去使用public继承表现出来的is-a关系,而是使用“拥有”这样的has-a关系。
复合的目的就是在Person类里面可以很自然地操作Person的属性,比如输出HomeNumber,查询HomeNumber等,这些可以调用PhoneNumber里面现成的成员函数。标准库里的string类型更是集成了丰富的成员方法,可以方便我们对Name进行各种各样的操作。
除了has-a关系外,复合还有一种含义,那就是is-implemented-in-terms-of,这说起来有些长,中文的意思是据某物实现出。举书上的例子,假定我们需要利用list来实现set,如果采用的是继承,像这样:
class MySet: public list
{
…
}
那就麻烦了,还记得我们在最初讲public继承时,就说了,public继承链下有Liskov法则,即父类存在的地方一定可以被子类所替代。这个例子里父类是list,子类是MySet,如果用MySet去替换list,那么就会有问题,因为list是支持重复元素的,如果连续执行两次list.push_back(1),那么list里面会有两个1元素,但换成MySet,结果却只会有一个1元素(因为Set是不重复元素的集合)。那怎么办呢,我们既想利用list现有的特性,也不想违反可替代法则。
方法就是复合,像下面这样:
template <class T>
class MySet
{
list<T> MyList;
…
}
这样就可以放心使用list现有的方法去实现MySet的功能了,具体的示例代码如下:
#include <iostream>
#include <list>
using namespace std;
template <class T>
class MySet
{
private:
list<T> MyList; public:
int Size() const
{
return MyList.size();
} bool IsContained(T Element) const
{
return (find(MyList.begin(), MyList.end(), T) != MyList.end());
} bool Insert(T Element)
{
if (!IsContained(T))
{
MyList.push_back(Element);
return true;
}
else
{
return false;
}
} bool Remove(T Element)
{
list<T>::iterator Iter = find(MyList.begin(), MyList.end(), T);
if (Iter != MyList.end())
{
MyList.erase(Iter);
return true;
}
else
{
return false;
}
}
};
好,到目前为止,大家应该能理解什么叫is-implemented-in-terms-of了吧,就是依据某物来实现,就像这里的set,就是依据于list来实现自身的结构的,这种情况下也用has-a复合模型。
最后总结一下:
1. 复合的意义和public继承完全不同;
2. 在应用域,复合意味着has-a(有一个),在实现域,复合意味着is-implemented-in-terms-of(根据某物实现出)。
读书笔记_Effective_C++_条款三十八:通过复合塑模出has-a或者is-implemented-in-terms-of的更多相关文章
- 读书笔记_Effective_C++_条款三十九:明智而审慎地使用private继承
private继承的意义在于“be implemented in turns of”,这个与上一条款中说的复合模型的第二层含义是相同的,这也意味着通常我们可以在这两种设计方法之间转换,但书上还是更提倡 ...
- 读书笔记_Effective_C++_条款三十:了解inline的里里外外
学过基本程序课的同学都知道,inline是内联的关键字,它可以建议编译器将函数的每一个调用都用函数本体替换.这是一种以空间换时间的做法.把每一次调用都用本体替换,无疑会使代码膨胀,但可以节省函数调用的 ...
- 读书笔记_Effective_C++_条款三十四:区分接口继承和实现继承
这个条款书上内容说的篇幅比较多,但其实思想并不复杂.只要能理解三句话即可,第一句话是:纯虚函数只继承接口:第二句话是:虚函数既继承接口,也提供了一份默认实现:第三句话是:普通函数既继承接口,也强制继承 ...
- 读书笔记_Effective_C++_条款三十二:确定你的public继承继承塑模出is-a关系
这一条款是说的是公有继承的逻辑,如果使用继承,而且继承是公有继承的话,一定要确保子类是一种父类(is-a关系).这种逻辑可能与生活中的常理不相符,比如企鹅是生蛋的,所有企鹅是鸟类的一种,直观来看,我们 ...
- 读书笔记_Effective_C++_条款四十八:了解模板元编程
作为模板部分的结束节,本条款谈到了模板元编程,元编程本质上就是将运行期的代价转移到编译期,它利用template编译生成C++源码,举下面阶乘例子: template <int N> st ...
- 读书笔记_Effective_C++_条款三十六:绝不重新定义继承而来的non-virtual函数
这个条款的内容很简单,见下面的示例: class BaseClass { public: void NonVirtualFunction() { cout << "BaseCla ...
- 读书笔记_Effective_C++_条款二十八:避免返回handlers指向对象内部成分
举个例子: class Student { private: int ID; string name; public: string& GetName() { return name; } } ...
- 读书笔记_Effective_C++_条款三十五:考虑virtual函数以外的其他选择
举书上的例子,考虑一个virtual函数的应用实例: class GameCharacter { private: int BaseHealth; public: virtual int GetHea ...
- 读书笔记_Effective_C++_条款二十九:为“异常安全”而努力是值得的
还是举书上的例子: void PrettyMenu::changeBackground(std::istream& imgSrc) { lock(&mutex); delete bgI ...
随机推荐
- Python模块制作
在Python中,每个Python文件都可以作为一个模块,模块的名字就是文件的名字. 定义自己的模块 比如有这样一个文件test.py,在test.py中定义了函数add def add(a,b): ...
- sublime Text快捷键(超级全)
sublime Text快捷键(超级全) Ctrl+Shift+P:打开命令面板 Ctrl+P:搜索项目中的文件 Ctrl+G:跳转到第几行 Ctrl+W:关闭当前打开文件 Ctrl+Shift+W: ...
- Oracle 中count(1) 、count(*) 和count(列名) 函数的区别
1)count(1)与count(*)比较: 1.如果你的数据表没有主键,那么count(1)比count(*)快2.如果有主键的话,那主键(联合主键)作为count的条件也比count(*)要快3. ...
- 关于整型Integer、Int32、Int64、IntPtr、UINT、UInt32、Cardinal、UInt64、UIntPtr、NativeUInt、Pointer、Handle
知识点1:UIntPtr = NativeUInt = Pointer = Handle 随程序的位数改变而改变.如下: 所以以后再用指针的时候要这样:UintPtr/NativeUInt(实例) = ...
- UVA10212 【The Last Non-zero Digit.】
暴力可做!!!(十秒还不打暴力!!!)暴力算阶乘边算边取余上代码 #include<iostream> #define int long long //开long long using n ...
- USACO 6.4 Wisconsin Squares
Wisconsin Squares It's spring in Wisconsin and time to move the yearling calves to the yearling past ...
- shell如何向python传递参数,shell如何接受python的返回值
1.shell如何向python传递参数 shell脚本 python $sendmailCommandPath $optDate python脚本 lastDateFormat = sys.argv ...
- [CodeForces - 848B] Rooter's Song 思维 找规律
大致题意: 有一个W*H的长方形,有n个人,分别站在X轴或Y轴,并沿直线向对面走,第i个人在ti的时刻出发,如果第i个人与第j个人相撞了 那么则交换两个人的运动方向,直到走到长方形边界停止,问最后每个 ...
- WebLogic和Tomcat的区别
J2ee开发主要是浏览器和服务器进行交互的一种结构.逻辑都是在后台进行处理,然后再把结果传输回给浏览器.可以看出服务器在这种架构是非常重要的. 这几天接触到两种Java的web服务器,做项目用的Tom ...
- SQL注入备忘录
备忘录(一) 拿起小本本记下常考知识点. 常用连接词 and && %23%23 且 or || %7c%7c 或 xor 非 Access 数据库: 只能爆破表名.列名获取数据.无法 ...