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)