读书笔记_Effective_C++_条款三十八:通过复合塑模出has-a或者is-implemented-in-terms-of
2014-03-15 00:11
615 查看
如果说public是一种is-a的关系的话,那么复合就是has-a的关系。直观来说,复合就是在一个类中采用其他类的对象作为自身的成员变量,可以举个例子,像下面这样:
我们一般会说人有名字,有家庭电话,有手机电话等,但我们一般不会说人是一个名字,或者人是一个家庭电话等。所以在这里,我们并不会去使用public继承表现出来的is-a关系,而是使用“拥有”这样的has-a关系。
复合的目的就是在Person类里面可以很自然地操作Person的属性,比如输出HomeNumber,查询HomeNumber等,这些可以调用PhoneNumber里面现成的成员函数。标准库里的string类型更是集成了丰富的成员方法,可以方便我们对Name进行各种各样的操作。
除了has-a关系外,复合还有一种含义,那就是is-implemented-in-terms-of,这说起来有些长,中文的意思是据某物实现出。举书上的例子,假定我们需要利用list来实现set,如果采用的是继承,像这样:
那就麻烦了,还记得我们在最初讲public继承时,就说了,public继承链下有Liskov法则,即父类存在的地方一定可以被子类所替代。这个例子里父类是list,子类是MySet,如果用MySet去替换list,那么就会有问题,因为list是支持重复元素的,如果连续执行两次list.push_back(1),那么list里面会有两个1元素,但换成MySet,结果却只会有一个1元素(因为Set是不重复元素的集合)。那怎么办呢,我们既想利用list现有的特性,也不想违反可替代法则。
方法就是复合,像下面这样:
这样就可以放心使用list现有的方法去实现MySet的功能了,具体的示例代码如下:
好,到目前为止,大家应该能理解什么叫is-implemented-in-terms-of了吧,就是依据某物来实现,就像这里的set,就是依据于list来实现自身的结构的,这种情况下也用has-a复合模型。
最后总结一下:
1. 复合的意义和public继承完全不同;
2. 在应用域,复合意味着has-a(有一个),在实现域,复合意味着is-implemented-in-terms-of(根据某物实现出)。
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++ Item 38 通过组合(composition)为 “has-a”或者“is-implemented-in-terms-of”建模
- Effective C++ Item 38 通过复合塑模出 has-a 或 is-implemented-in-terms-of
- Effective C++ Item 38 通过复合塑模出 has-a 或 is-implemented-in-terms-of
- 条款38:通过聚合设计has-a或者is-implemented-in-terms-of
- [翻译] Effective C++, 3rd Edition, Item 38: 通过 composition(复合)模拟 "has-a"(有一个)或 "is-implemented-in-terms-of"(是根据……实现的)
- Item 38: Model "has-a" or "is-implemented-in-terms-of" through composition(Effective C++)
- 通过 composition模拟 "has-a"或 "is-implemented-in-terms-of"
- Effective C++ 38. Model "has-a or is-implemented-in-terms-of" through composition
- 读书笔记_Effective_C++_条款三十二:确定你的public继承继承塑模出is-a关系
- C++出现to_string is not a member of std 或者 to_string was not declared in this scope的解决方法
- 读书笔记《Effective C++》条款38:通过复合塑模出has-a或“根据某物实现出”
- 读书笔记_Effective_C++_条款四十九:了解new_handler的行为
- 读书笔记_Effective_C++_条款三十七:绝不重新定义继承而来的缺省参数值
- 《MORE EFFECTIVE C++》条款27 要求或者禁止对象分配在堆上
- 读书笔记_Effective_C++_条款四十五:运用成员函数模板接受所有兼容类型
- 读书笔记_Effective_C++_条款十八:让接口容易被正确使用,不易被误用
- <<Effective c++>>读书笔记---条款19:设计class犹如设计type
- 出现 Assigning the return value of new by reference is deprecated in xxxx &&“Warning: Call-time pass-by-reference has been deprecated”怎么办?
- 读书笔记_Effective_C++_条款六:若不想使用编译器自动生成的函数,就该明确拒绝
- kingofark's Ineffective C/C++:自白1:返回值的运用