Bridge定義:將抽象和行為劃分開(kāi)來(lái),各自獨(dú)立,但能動(dòng)態(tài)的結(jié)合。
為什么使用橋模式
通常,當(dāng)一個(gè)抽象類(lèi)或接口有多個(gè)具體實(shí)現(xiàn)(concrete subclass),這些concrete之間關(guān)系可能有以下兩種:
1.這多個(gè)具體實(shí)現(xiàn)之間恰好是并列的,如前面舉例,打樁,有兩個(gè)concrete class:方形樁和圓形樁;這兩個(gè)形狀上的樁是并列的,沒(méi)有概念上的重復(fù),那么我們只要使用繼承就可以了。
2.實(shí)際應(yīng)用上,常常有可能在這多個(gè)concrete class之間有概念上重疊。那么需要我們把抽象共同部分和行為共同部分各自獨(dú)立開(kāi)來(lái),原來(lái)是準(zhǔn)備放在一個(gè)接口里,現(xiàn)在需要設(shè)計(jì)兩個(gè)接口,分別放置抽象和行為。
例如,一杯咖啡為例,有中杯和大杯之分,同時(shí)還有加奶 不加奶之分。如果用單純的繼承,這四個(gè)具體實(shí)現(xiàn)(中杯 大杯 加奶 不加奶)之間有概念重疊,因?yàn)橛兄斜幽蹋灿兄斜患幽蹋绻僭谥斜@一層再實(shí)現(xiàn)兩個(gè)繼承,很顯然混亂,擴(kuò)展性極差。那我們使用Bridge模式來(lái)實(shí)現(xiàn)它。
如何實(shí)現(xiàn)橋模式
以上面提到的咖啡 為例。我們?cè)瓉?lái)打算只設(shè)計(jì)一個(gè)接口(抽象類(lèi)),使用Bridge模式后,我們需要將抽象和行為分開(kāi),加奶和不加奶屬于行為,我們將它們抽象成一個(gè)專(zhuān)門(mén)的行為接口。
先看看抽象部分的接口代碼:
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();
}
現(xiàn)在我們有了兩個(gè)抽象類(lèi),下面我們分別對(duì)其進(jìn)行繼承,實(shí)現(xiàn)concrete class:
//中杯
public class MediumCoffee extends Coffee{
public MediumCoffee() {setCoffeeImp();}
public void pourCoffee(){
CoffeeImp coffeeImp = this.getCoffeeImp();
//我們以重復(fù)次數(shù)來(lái)說(shuō)明是沖中杯還是大杯 ,重復(fù)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();
//我們以重復(fù)次數(shù)來(lái)說(shuō)明是沖中杯還是大杯 ,重復(fù)5次是大杯
for (int i = 0; i < 5; i++){
coffeeImp.pourCoffeeImp();
}
}
}
上面分別是中杯和大杯的具體實(shí)現(xiàn).下面再對(duì)行為CoffeeImp進(jìn)行繼承:
//加奶
public class MilkCoffeeImp extends CoffeeImp{
MilkCoffeeImp() {}
public void pourCoffeeImp(){
System.out.println("加了美味的牛奶");
}
}
//不加奶
public class FragrantCoffeeImp extends CoffeeImp{
FragrantCoffeeImp() {}
public void pourCoffeeImp(){
System.out.println("什么也沒(méi)加,清香");
}
}
Bridge模式的基本框架我們已經(jīng)搭好了,別忘記定義中還有一句:動(dòng)態(tài)結(jié)合,我們現(xiàn)在可以喝到至少四種咖啡:
1.中杯加奶
2.中杯不加奶
3.大杯加奶
4.大杯不加奶
看看是如何動(dòng)態(tài)結(jié)合的,在使用之前,我們做個(gè)準(zhǔn)備工作,設(shè)計(jì)一個(gè)單態(tài)類(lèi)(Singleton)用來(lái)hold當(dāng)前的CoffeeImp:
public class CoffeeImpSingleton{
private static CoffeeImp coffeeImp;
public CoffeeImpSingleton(CoffeeImp coffeeImpIn)
{this.coffeeImp = coffeeImpIn;}
public static CoffeeImp getTheCoffeeImp(){
return coffeeImp;
}
}
看看中杯加奶 和大杯加奶 是怎么出來(lái)的:
//拿出牛奶
CoffeeImpSingleton coffeeImpSingleton = new CoffeeImpSingleton(new MilkCoffeeImp());
//中杯加奶
MediumCoffee mediumCoffee = new MediumCoffee();
mediumCoffee.pourCoffee();
//大杯加奶
SuperSizeCoffee superSizeCoffee = new SuperSizeCoffee();
superSizeCoffee.pourCoffee();
注意:Bridge模式的執(zhí)行類(lèi)如CoffeeImp和Coffee是一對(duì)一的關(guān)系,正確創(chuàng)建CoffeeImp是該模式的關(guān)鍵。
Bridge模式在EJB中的應(yīng)用
EJB中有一個(gè)Data Access Object (DAO)模式,這是將商業(yè)邏輯和具體數(shù)據(jù)資源分開(kāi)的,因?yàn)椴煌臄?shù)據(jù)庫(kù)有不同的數(shù)據(jù)庫(kù)操作。將操作不同數(shù)據(jù)庫(kù)的行為獨(dú)立抽象成一個(gè)行為接口DAO,如下:
1.Business Object (類(lèi)似Coffee)
實(shí)現(xiàn)一些抽象的商業(yè)操作:如尋找一個(gè)用戶(hù)下所有的訂單。涉及數(shù)據(jù)庫(kù)操作都使用DAOImplementor。
2.Data Access Object (類(lèi)似CoffeeImp)
一些抽象的對(duì)數(shù)據(jù)庫(kù)資源操作。
3.DAOImplementor 如OrderDAOCS, OrderDAOOracle, OrderDAOSybase(類(lèi)似MilkCoffeeImp FragrantCoffeeImp)
具體的數(shù)據(jù)庫(kù)操作,如"INSERT INTO "等語(yǔ)句,OrderDAOOracle是Oracle OrderDAOSybase是Sybase數(shù)據(jù)庫(kù)。
4.數(shù)據(jù)庫(kù) (Cloudscape, Oracle, or Sybase database via JDBC API)