概念
盡量使用合成/聚合,而不是使用繼承實作複用。所謂的合成/聚合是指一個對象裡持有另外一個類的對象,通過調用這些對象的方法得到複用已有功能的目的。如:封包解譯程式中,按照繼承複用可以設計為:
子類調用父類的方法即可完成水文封包解譯、氣象解譯中通用方法;子類中一定包含了父類的方法,這個叫繼承複用。
按照合成/聚合原則設計為:
水文協定和氣象協定中,持有編碼和位制轉換對象,通過調用對象方法即可完成複用。
示例
資料庫連接配接的複用:首先看通過內建關系複用資料連接配接代碼如下
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、資料位制轉換,就沒辦法使用。