继承是类与类或者接口与接口之间最常见的一种关系;继承是一种is-a关系。
组合(Composition)体现的是整体与部分、拥有的关系,即has-a的关系。
is-a:表示"是一个"的关系,如狗是一个动物
has-a:表示"有一个"的关系,如狗有一个尾巴
在继承
结构中,父类的内部细节对于子类是可见的。所以我们通常也可以说通过继承的代码复用是一种白盒式代码复用
。(如果基类的实现发生改变,那么派生类的实现也将随之改变。这样就导致了子类行为的不可预知性;)
组合
是通过对现有的对象进行拼装(组合)产生新的、更复杂的功能。因为在对象之间,各自的内部细节是不可见的,所以我们也说这种方式的代码复用是黑盒式代码复用
。(因为组合中一般都定义一个类型,所以在编译期根本不知道具体会调用哪个实现类的方法)
继承
,在写代码的时候就要指名具体继承哪个类,所以,在编译期
就确定了关系。(从基类继承来的实现是无法在运行期动态改变的,因此降低了应用的灵活性。)
组合
,在写代码的时候可以采用面向接口编程。所以,类的组合关系一般在运行期
确定。