天天看点

面向对象设计原则——合成复用原则

概念

尽量使用合成/聚合,而不是使用继承实现复用。所谓的合成/聚合是指一个对象里持有另外一个类的对象,通过调用这些对象的方法得到复用已有功能的目的。如:报文解译程序中,按照继承复用可以设计为:

面向对象设计原则——合成复用原则

子类调用父类的方法即可完成水文报文解译、气象解译中通用方法;子类中一定包含了父类的方法,这个叫继承复用。

按照合成/聚合原则设计为:

面向对象设计原则——合成复用原则

水文协议和气象协议中,持有编码和位制转换对象,通过调用对象方法即可完成复用。

示例

数据库连接的复用:首先看通过集成关系复用数据连接代码如下

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、数据位制转换,就没办法使用。

继续阅读