Java橋模式(Bridge模式)

技術小能手發表於2018-08-22

為什麼使用橋模式

通常,當一個抽象類或介面有多個具體實現(concrete subclass),這些concrete之間關係可能有以下兩種:

這多個具體實現之間恰好是並列的,如前面舉例,打樁,有兩個concrete class:方形樁和圓形樁;這兩個形狀上的樁是並列的,沒有概念上的重複,那麼我們只要使用繼承就可以了。

實際應用上,常常有可能在這多個concrete class之間有概念上重疊。那麼需要我們把抽象共同部分和行為共同部分各自獨立開來,原來是準備放在一個介面裡,現在需要設計兩個介面,分別放置抽象和行為。
例如,一杯咖啡為例,有中杯和大杯之分,同時還有加奶 不加奶之分。如果用單純的繼承,這四個具體實現(中杯 大杯 加奶 不加奶)之間有概念重疊,因為有中杯加奶,也有中杯不加奶,如果再在中杯這一層再實現兩個繼承,很顯然混亂,擴充套件性極差。那我們使用Bridge模式來實現它。

如何實現橋模式

以上面提到的咖啡 為例。我們原來打算只設計一個介面(抽象類),使用Bridge模式後,我們需要將抽象和行為分開,加奶和不加奶屬於行為,我們將它們抽象成一個專門的行為介面。

先看看抽象部分的介面程式碼:


  1. public abstract class Coffee{

  2. CoffeeImp coffeeImp;

  3. public void setCoffeeImp() {

  4. this.CoffeeImp = CoffeeImpSingleton.getTheCoffeImp();

  5. }

  6. public CoffeeImp getCoffeeImp() {return this.CoffeeImp;}

  7. public abstract void pourCoffee();

  8. }

其中CoffeeImp 是加不加奶的行為介面,看其程式碼如下:


  1. public abstract class CoffeeImp{

  2. public abstract void pourCoffeeImp();

  3. }

現在我們有了兩個抽象類,下面我們分別對其進行繼承,實現concrete class:


  1. //中杯

  2. public class MediumCoffee extends Coffee{

  3. public MediumCoffee() {setCoffeeImp();}

  4. public void pourCoffee(){

  5. CoffeeImp coffeeImp = this.getCoffeeImp();

  6. //我們以重複次數來說明是衝中杯還是大杯 ,重複2次是中杯

  7. for (int i = 0; i < 2; i++){

  8.    coffeeImp.pourCoffeeImp();

  9. }

  10. }

  11. }


  1. //大杯

  2. public class SuperSizeCoffee extends Coffee{

  3. public SuperSizeCoffee() {setCoffeeImp();}

  4. public void pourCoffee(){

  5. CoffeeImp coffeeImp = this.getCoffeeImp();

  6. //我們以重複次數來說明是衝中杯還是大杯 ,重複5次是大杯

  7. for (int i = 0; i < 5; i++){

  8.    coffeeImp.pourCoffeeImp();

  9. }

  10. }

  11. }

上面分別是中杯和大杯的具體實現.下面再對行為CoffeeImp進行繼承:


  1. //加奶

  2. public class MilkCoffeeImp extends CoffeeImp{

  3. MilkCoffeeImp() {}

  4. public void pourCoffeeImp(){

  5. System.out.println("加了美味的牛奶");

  6. }

  7. }

  8. //不加奶

  9. public class FragrantCoffeeImp extends CoffeeImp{

  10. FragrantCoffeeImp() {}

  11. public void pourCoffeeImp(){

  12. System.out.println("什麼也沒加,清香");

  13. }

  14. }

Bridge模式的基本框架我們已經搭好了,別忘記定義中還有一句:動態結合,我們現在可以喝到至少四種咖啡:

中杯加奶
中杯不加奶
大杯加奶
大杯不加奶
看看是如何動態結合的,在使用之前,我們做個準備工作,設計一個單態類(Singleton)用來hold當前的CoffeeImp:


  1. public class CoffeeImpSingleton{

  2. private static CoffeeImp coffeeImp;

  3. public CoffeeImpSingleton(CoffeeImp coffeeImpIn)

  4. {this.coffeeImp = coffeeImpIn;}

  5. public static CoffeeImp getTheCoffeeImp(){

  6. return coffeeImp;

  7. }

  8. }

看看中杯加奶 和大杯加奶 是怎麼出來的:


  1. //拿出牛奶

  2. CoffeeImpSingleton coffeeImpSingleton = new CoffeeImpSingleton(new MilkCoffeeImp());

  3. //中杯加奶

  4. MediumCoffee mediumCoffee = new MediumCoffee();

  5. mediumCoffee.pourCoffee();

  6. //大杯加奶

  7. SuperSizeCoffee superSizeCoffee = new SuperSizeCoffee();

  8. superSizeCoffee.pourCoffee();

注意:Bridge模式的執行類如CoffeeImp和Coffee是一對一的關係,正確建立CoffeeImp是該模式的關鍵。

Bridge模式在EJB中的應用

EJB中有一個Data Access Object (DAO)模式,這是將商業邏輯和具體資料資源分開的,因為不同的資料庫有不同的資料庫操作。將操作不同資料庫的行為獨立抽象成一個行為介面DAO,如下:

  • Business Object (類似Coffee)

實現一些抽象的商業操作:如尋找一個使用者下所有的訂單。涉及資料庫操作都使用DAOImplementor。

  • Data Access Object (類似CoffeeImp)

一些抽象的對資料庫資源操作。

  • DAOImplementor 如OrderDAOCS, OrderDAOOracle, OrderDAOSybase(類似MilkCoffeeImp FragrantCoffeeImp)

具體的資料庫操作,如”INSERT INTO “等語句,OrderDAOOracle是Oracle OrderDAOSybase是Sybase資料庫。

  • 資料庫 (Cloudscape, Oracle, or Sybase database via JDBC API)

原文釋出時間為:2018-08-21

本文作者:HARRIES

本文來自雲棲社群合作伙伴“Java雜記”,瞭解相關資訊可以關注“Java雜記”。


相關文章