定义:就是说要尽量的使用合成和聚合,而不是继承关系达到复用的目的
该原则就是在一个新的对象里面使用一些已有的对象,使之成为新对象的一部分:新的对象通过向这些对象的委派达到复用已有功能的目的
1、继承复用破坏了类的封装性。因为继承会将基类的实现细节暴露给派生类,基类对派生类是透明的,所以这种复用又称为 “白箱” 复用
2、派生类与基类的耦合度高。基类的实现的任何改变都会导致派生类的实现发生变化,这不利于类的扩展与维护
3、它限制了复用的灵活性。从基类继承而来的实现是静态的,在编译时已经定义,所以运行时不可能发生变化
4、当复用派生类的时候,如果继承下来的实现不适合解决新的问题,则基类必须重写或者被其它更适合的类所替换,这种依赖关系限制了灵活性,最终限制了复用性
1、维持了类的封装性。因为成分对象的内部细节是新对象看不见的,所以这种复用又称为 “黑箱” 复用
2、新旧类之间的耦合度低。这种复用所需的依赖较少,新对象存取成分对象的唯一方法是通过成分对象的接口
3、复用的灵活性高。这种复用可以在运行时动态进行,新对象可以动态地将新的责任委派到合适的对象
4、每一个新的类可以将焦点集中在一个任务上
反例代码 v1 版本
class MySet extends HashSet { private int count = 0; public boolean add(Object obj) { count++; return super.add(obj); } @Override public boolean addAll(Collection c) { count += c.size(); return super.addAll(c); } public int getCount() { return count; } }
public class Client { public static void main(String[] args) { Set set2 = new HashSet(); set2.add("Java性能优化权威指南"); MySet set = new MySet(); set.addAll(set2); System.out.println(set.getCount()); } }
最终输出 2,因为基类的 addAll 回调了 add 方法
反例代码 v2 版本,由于 addAll 会回调 add 方法,导致累加不正确,修改代码:MySet 类不重写 addAll,client 类不变
class MySet extends HashSet { private int count = 0; public boolean add(Object obj) { count++; return super.add(obj); } public int getCount() { return count; } }
看似好像没有问题了,但是目前的代码必须依赖于 HashSet 的 addAll 方法必须回调 add 方法,如果 JDK 版本升级,addAll 不在回调 add 方法,那么自定义的 MySet 将会出问题,依赖性太强
反例代码 v3 版本,针对 v2 版本的问题,修改如下:MySet 重写 addAll,不再做 count + c.size() 操作,而是保证 addAll 一定回调 add 方法
class MySet extends HashSet { private int count = 0; public boolean add(Object obj) { count++; return super.add(obj); } @Override public boolean addAll(Collection c) { boolean bln = false; for(Object obj : c) { if (add(c)) { bln = true; } } return bln; } public int getCount() { return count; } }
最终输出 4,看似好像没有问题了,但其实又有问题,始终围绕着 HashSet 在转,依赖性太强
1、万一 JDK 更新(好比 HashMap 底层数据结构每个版本不太一样),HashSet 多了入口方法 addOne(),MySet 没有重写,会导致程序错误
2、目前重写了 add、addAll 两个方法,万一在 HashSet 中有方法依赖于这两个方法,会导致业务错误
最终完美的 v4 版本,MySet 不在继承 HashSet,让 MySet 与 HashSet 发生关联关系
class MySet { private Set set = new HashSet(); private int count = 0; public boolean add(Object obj) { count++; return set.add(obj); } public boolean addAll(Collection c) { count += c.size(); return set.addAll(c); } public int getCount() { return count; } }
JDK 的反例教材,为了复用 remove、get、put 方法,Stack extends Vector,从而导致了栈不是栈
public class Client { public static void main(String[] args) { Stack<String> stack = new Stack<>(); // 入栈出栈 FILO 先进后出 // 入栈 stack.push("A"); stack.push("B"); // 出栈 System.out.println(stack.pop()); // 输出 B System.out.println(stack.pop()); // 输出 A System.out.println(stack.remove(0)); // 输出 A,这就不是先进后出 System.out.println(stack.get(0)); // 输出 A,这就不是先进后出 } }
反例教材不代表以后不在使用继承、方法重写。是否使用继承取决于基类与派生类的作者是否为同一人:因为如果不是同一人,那么基类作者不知道也不会管派生类重写了什么方法,而派生类也预知不了基类未来会增加什么方法
如果只是为了复用代码,应当使用组合关系,组合大于继承。使用继承关系,难免会出现问题。如果要使用继承关系,则必须严格遵循里氏替换原则。合成复用原则和里氏替换原则相辅相成的,两者都是开闭原则的具体实现规范