概念
尽量使用合成/聚合,而不是使用继承实现复用。所谓的合成/聚合是指一个对象里持有另外一个类的对象,通过调用这些对象的方法得到复用已有功能的目的。如:报文解译程序中,按照继承复用可以设计为:
子类调用父类的方法即可完成水文报文解译、气象解译中通用方法;子类中一定包含了父类的方法,这个叫继承复用。
按照合成/聚合原则设计为:
水文协议和气象协议中,持有编码和位制转换对象,通过调用对象方法即可完成复用。
示例
数据库连接的复用:首先看通过集成关系复用数据连接代码如下
public class SqlServerConnect { private Connection con = null; public Connection getCon() { System.out.println("创建数据库连接"); return con; } } public class UserDao extends SqlServerConnect { //继承复用连接数据 public void queryData() { Connection con =getCon(); String sql = "select * from emp"; try { Statement statement = con.createStatement(); } catch (SQLException e) { e.printStackTrace(); } } }
UserDao继承了SqlServerConnect,复用了父类的getCon()方法;如果此时数据库不再使用SQLServer,要改成oracle,这种复用就无能为力了。
使用合成复用,代码可以修改为:
//增加一个接口 public interface DatabaseConnection { Connection getCon(); } //SqlServerConnect实现该接口 public class SqlServerConnect implements DatabaseConnection { private Connection con = null; @Override public Connection getCon() { System.out.println("创建数据库连接"); return con; } } //clsUserDaoNew和数据库连接接口呈现聚合关系,使用依赖倒置,可动态替换此类,复用了getCon()代码 public class UserDaoNew { private DatabaseConnection objCon; public UserDaoNew(DatabaseConnection conn){ objCon = conn; } public void queryData() { Connection con = objCon.getCon(); String sql = "select * from emp"; try { Statement statement = con.createStatement(); } catch (SQLException e) { e.printStackTrace(); } } }
在UserDaoNew类中,通过使用聚合关系,通过构造注入一个数据连接对象,通过调用这个对象的getCon()方法实现复用。这种方式,利用里氏代换和依赖倒置原则,当使用SQLServer数据库时,注入SqlServerConnect实例,如果使用oracle数据库时,注入OrcaleConnect实例,代码更加灵活,实现动态复用。
拓展
1.继承是静态复用,通过聚合复用是动态复用。所谓的静态复用是在编码阶段已经明确了类之间的关系;动态复用则是在程序运行阶段,根据实际要求注入相应的对象完成复用的,动态复用比静态复用更具有灵活性。
2.合成复用原则还体现复用范围扩大了。如上图所示,使用继承关系,则BCD转ASCII码只服务报文解译,如一个加密程序也要使用BCD转ASCII、数据位制转换,就没办法使用。
以上就是java面向对象设计原则之合成复用原则示例详解的详细内容,更多关于java面向对象设计原则的资料请关注服务器之家其它相关文章!
原文链接:https://blog.csdn.net/guoyp2126/article/details/113946537