概念
尽量使用合成/聚合,而不是使用继承实现复用。所谓的合成/聚合是指一个对象里持有另外一个类的对象,通过调用这些对象的方法得到复用已有功能的目的。如:报文解译程序中,按照继承复用可以设计为:
子类调用父类的方法即可完成水文报文解译、气象解译中通用方法;子类中一定包含了父类的方法,这个叫继承复用。
按照合成/聚合原则设计为:
水文协议和气象协议中,持有编码和位制转换对象,通过调用对象方法即可完成复用。
示例
数据库连接的复用:首先看通过集成关系复用数据连接代码如下
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实例,代码更加灵活,实现动态复用。
拓展
- 继承是静态复用,通过聚合复用是动态复用。所谓的静态复用是在编码阶段已经明确了类之间的关系;动态复用则是在程序运行阶段,根据实际要求注入相应的对象完成复用的,动态复用比静态复用更具有灵活性。
- 合成复用原则还体现复用范围扩大了。如上图所示,使用继承关系,则BCD转ASCII码只服务报文解译,如一个加密程序也要使用BCD转ASCII、数据位制转换,就没办法使用。