天天看點

面向對象設計原則——合成複用原則

概念

盡量使用合成/聚合,而不是使用繼承實作複用。所謂的合成/聚合是指一個對象裡持有另外一個類的對象,通過調用這些對象的方法得到複用已有功能的目的。如:封包解譯程式中,按照繼承複用可以設計為:

面向對象設計原則——合成複用原則

子類調用父類的方法即可完成水文封包解譯、氣象解譯中通用方法;子類中一定包含了父類的方法,這個叫繼承複用。

按照合成/聚合原則設計為:

面向對象設計原則——合成複用原則

水文協定和氣象協定中,持有編碼和位制轉換對象,通過調用對象方法即可完成複用。

示例

資料庫連接配接的複用:首先看通過內建關系複用資料連接配接代碼如下

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、資料位制轉換,就沒辦法使用。

繼續閱讀