天天看點

設計模式之Bridge

Bridge定義 :

将抽象和行為劃分開來,各自獨立,但能動态的結合.

為什麼使用?

通常,當一個抽象類或接口有多個具體實作(concrete subclass),這些concrete之間關系可能有以下兩種:

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

2.實際應用上,常常有可能在這多個concrete class之間有概念上重疊.那麼需要我們把抽象共同部分和行為共同部分各自獨立開來,原來是準備放在一個接口裡,現在需要設計兩個接口,分别放置抽象和行為.

例如,一杯咖啡為例,有中杯和大杯之分,同時還有加奶 不加奶之分. 如果用單純的繼承,這四個具體實作(中杯 大杯 加奶 不加奶)之間有概念重疊,因為有中杯加奶,也有中杯不加奶, 如果再在中杯這一層再實作兩個繼承,很顯然混亂,擴充性極差.那我們使用Bridge模式來實作它.

如何實作?

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

先看看抽象部分的接口代碼:

public abstract class Coffee

{

  CoffeeImp coffeeImp;

  public void setCoffeeImp() {

    this.CoffeeImp = CoffeeImpSingleton.getTheCoffeImp();

  }

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

  public abstract void pourCoffee();

}

其中CoffeeImp 是加不加奶的行為接口,看其代碼如下:

public abstract class CoffeeImp

{

  public abstract void pourCoffeeImp();

}

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

//中杯

public class MediumCoffee extends Coffee

{

  public MediumCoffee() {setCoffeeImp();}

  public void pourCoffee()

  {

    CoffeeImp coffeeImp = this.getCoffeeImp();

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

    for (int i = 0; i < 2; i++)

    {

      coffeeImp.pourCoffeeImp();

    }

  

  }

}

//大杯

public class SuperSizeCoffee extends Coffee

{

  public SuperSizeCoffee() {setCoffeeImp();}

  public void pourCoffee()

  {

    CoffeeImp coffeeImp = this.getCoffeeImp();

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

    for (int i = 0; i < 5; i++)

    {

      coffeeImp.pourCoffeeImp();

    }

  

  }

}

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

//加奶

public class MilkCoffeeImp extends CoffeeImp

{

  MilkCoffeeImp() {}

  public void pourCoffeeImp()

  {

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

  }

}

//不加奶

public class FragrantCoffeeImp extends CoffeeImp

{

  FragrantCoffeeImp() {}

  public void pourCoffeeImp()

  {

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

  }

}

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

1.中杯加奶

2.中杯不加奶

3.大杯加奶

4.大杯不加奶

看看是如何動态結合的,在使用之前,我們做個準備工作,設計一個單态類(Singleton)用來hold目前的CoffeeImp:

public class CoffeeImpSingleton

{

  private static CoffeeImp coffeeImp;

  public CoffeeImpSingleton(CoffeeImp coffeeImpIn)

   {this.coffeeImp = coffeeImpIn;}

  public static CoffeeImp getTheCoffeeImp()

  {

    return coffeeImp;

  }

}

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

//拿出牛奶

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

//中杯加奶

MediumCoffee mediumCoffee = new MediumCoffee();

mediumCoffee.pourCoffee();

//大杯加奶

SuperSizeCoffee superSizeCoffee = new SuperSizeCoffee();

superSizeCoffee.pourCoffee();

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

Bridge模式在EJB中的應用

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

1.Business Object (類似Coffee)

實作一些抽象的商業操作:如尋找一個使用者下所有的訂單

涉及資料庫操作都使用DAOImplementor.

2.Data Access Object (類似CoffeeImp)

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

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

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

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