java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的 Java 程序设计语言和 Java 平台(即 JavaEE(j2ee), JavaME(j2me), JavaSE(j2se))的总称。
这篇文章主要介绍了合成聚合复用原则,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
合成聚合复用原则合成复用原则又称为组合 / 聚合复用原则 (Composition/Aggregate Reuse Principle, CARP),其定义如下:
合成复用原则 (Composite Reuse Principle, CRP):尽量使用对象组合,而不是继承来达到复用的目的。
合成复用原则就是在一个新的对象里通过关联关系(包括组合关系和聚合关系)来使用一些已有的对象,使之成为新对象的一部分;新对象通过委派调用已有对象的方法达到复用功能的目的。简言之:复用时要尽量使用组合 / 聚合关系(关联关系),少用继承。
在面向对象设计中,可以通过两种方法在不同的环境中复用已有的设计和实现,即通过组合 / 聚合关系或通过继承,但首先应该考虑使用组合 / 聚合,组合 / 聚合可以使系统更加灵活,降低类与类之间的耦合度,一个类的变化对其他类造成的影响相对较少;其次才考虑继承,在使用继承时,需要严格遵循里氏代换原则,有效使用继承会有助于对问题的理解,降低复杂度,而滥用继承反而会增加系统构建和维护的难度以及系统的复杂度,因此需要慎重使用继承复用。
通过继承来进行复用的主要问题在于继承复用会破坏系统的封装性,因为继承会将基类的实现细节暴露给子类,由于基类的内部细节通常对子类来说是可见的,所以这种复用又称 "白箱" 复用,如果基类发生改变,那么子类的实现也不得不发生改变;从基类继承而来的实现是静态的,不可能在运行时发生改变,没有足够的灵活性;而且继承只能在有限的环境中使用(如类没有声明为不能被继承)。
扩展对于继承的深入理解,大家可以参考《软件架构设计》一书作者温昱先生的文章——《见山只是山见水只是水——提升对继承的认识》。
由于组合或聚合关系可以将已有的对象(也可称为成员对象)纳入到新对象中,使之成为新对象的一部分,因此新对象可以调用已有对象的功能,这样做可以使得成员对象的内部实现细节对于新对象不可见,所以这种复用又称为 "黑箱" 复用,相对继承关系而言,其耦合度相对较低,成员对象的变化对新对象的影响不大,可以在新对象中根据实际需要有选择性地调用成员对象的操作;合成复用可以在运行时动态进行,新对象可以动态地引用与成员对象类型相同的其他对象。
一般而言,如果两个类之间是 "Has-A" 的关系应使用组合或聚合,如果是 "Is-A" 关系可使用继承。"Is-A" 是严格的分类学意义上的定义,意思是一个类是另一个类的 "一种";而 "Has-A" 则不同,它表示某一个角色具有某一项责任。
下面通过一个简单实例来加深对合成复用原则的理解:
Sunny 软件公司开发人员在初期的 CRM 系统设计中,考虑到客户数量不多,系统采用 MySQL 作为数据库,与数据库操作有关的类如 CustomerDAO 类等都需要连接数据库,连接数据库的方法 getConnection() 封装在 DBUtil 类中,由于需要重用 DBUtil 类的 getConnection() 方法,设计人员将 CustomerDAO 作为 DBUtil 类的子类,初始设计方案结构如图 1 所示:
图 1 初始设计方案结构图
随着客户数量的增加,系统决定升级为 Oracle 数据库,因此需要增加一个新的 OracleDBUtil 类来连接 Oracle 数据库,由于在初始设计方案中 CustomerDAO 和 DBUtil 之间是继承关系,因此在更换数据库连接方式时需要修改 CustomerDAO 类的源代码,将 CustomerDAO 作为 OracleDBUtil 的子类,这将违反开闭原则。【当然也可以修改 DBUtil 类的源代码,同样会违反开闭原则。】
现使用合成复用原则对其进行重构。
根据合成复用原则,我们在实现复用时应该多用关联,少用继承。因此在本实例中我们可以使用关联复用来取代继承复用,重构后的结构如图 2 所示:
图 2 重构后的结构图
在图 2 中,CustomerDAO 和 DBUtil 之间的关系由继承关系变为关联关系,采用依赖注入的方式将 DBUtil 对象注入到 CustomerDAO 中,可以使用构造注入,也可以使用 Setter 注入。如果需要对 DBUtil 的功能进行扩展,可以通过其子类来实现,如通过子类 OracleDBUtil 来连接 Oracle 数据库。由于 CustomerDAO 针对 DBUtil 编程,根据里氏代换原则,DBUtil 子类的对象可以覆盖 DBUtil 对象,只需在 CustomerDAO 中注入子类对象即可使用子类所扩展的方法。例如在 CustomerDAO 中注入 OracleDBUtil 对象,即可实现 Oracle 数据库连接,原有代码无须进行修改,而且还可以很灵活地增加新的数据库连接方式。
来源: http://www.phperz.com/article/18/0107/355496.html