適配器模式
定義:將一個(gè)接口轉(zhuǎn)換成客戶希望的另一個(gè)接口,適配器模式使接口不兼容的那些類可以一起工作,其別名為包裝器(Wrapper)。適配器模式既可以作為類結(jié)構(gòu)型模式,也可以作為對(duì)象結(jié)構(gòu)型模式。
適配器模式包含如下角色:
Target:目標(biāo)抽象類
Adapter:適配器類
Adaptee:適配者類
Client:客戶類
適配器模式有對(duì)象適配器和類適配器兩種實(shí)現(xiàn):
對(duì)象適配器
/**
* 對(duì)象適配器模式
*
*/
public class AdapterPattern {
/**
* 適配者類
*/
public static class Adaptee {
public void func1() {
System.out.println("func1....");
}
}
/**
* 適配器類
*/
public static class Adapter implements Target {
Adaptee adaptee;
public Adapter(Adaptee adaptee) {
this.adaptee = adaptee;
}
@Override
public void func1() {
adaptee.func1();
}
@Override
public void func2() {
System.out.println("func2...");
}
}
/**
* 目標(biāo)抽象類
*/
interface Target {
void func1();
void func2();
}
public static void main(String[] args) {
Adapter adapter = new Adapter(new Adaptee());
adapter.func1();
adapter.func2();
}
}
輸出
func1....
func2...
類適配器
package com.hzy.pattern;
/**
* 類適配器模式
*
*/
public class AdapterPattern {
/**
* 適配者類
*/
public static class Adaptee {
public void func1() {
System.out.println("func1....");
}
}
/**
* 適配器類
*/
public static class Adapter extends Adaptee implements Target {
@Override
public void func2() {
System.out.println("func2...");
}
}
/**
* 目標(biāo)抽象類
*/
interface Target {
void func1();
void func2();
}
public static void main(String[] args) {
Adapter adapter = new Adapter();
adapter.func1();
adapter.func2();
}
}
輸出
func1....
func2...
優(yōu)點(diǎn)
將目標(biāo)類和適配者類解耦,通過引入一個(gè)適配器類來重用現(xiàn)有的適配者類,而無須修改原有代碼。
增加了類的透明性和復(fù)用性,將具體的實(shí)現(xiàn)封裝在適配者類中,對(duì)于客戶端類來說是透明的,而且提高了適配者的復(fù)用性。
靈活性和擴(kuò)展性都非常好,通過使用配置文件,可以很方便地更換適配器,也可以在不修改原有代碼的基礎(chǔ)上增加新的適配器類,完全符合“開閉原則”。
類適配器模式還具有如下優(yōu)點(diǎn):
由于適配器類是適配者類的子類,因此可以在適配器類中置換一些適配者的方法,使得適配器的靈活性更強(qiáng)。
對(duì)象適配器模式還具有如下優(yōu)點(diǎn):
一個(gè)對(duì)象適配器可以把多個(gè)不同的適配者適配到同一個(gè)目標(biāo),也就是說,同一個(gè)適配器可以把適配者類和它的子類都適配到目標(biāo)接口。
缺點(diǎn)
類適配器模式的缺點(diǎn)如下:
對(duì)于Java、C#等不支持多重繼承的語言,一次最多只能適配一個(gè)適配者類,而且目標(biāo)抽象類只能為抽象類,不能為具體類,其使用有一定的局限性,不能將一個(gè)適配者類和它的子類都適配到目標(biāo)接口。
對(duì)象適配器模式的缺點(diǎn)如下:
與類適配器模式相比,要想置換適配者類的方法就不容易。如果一定要置換掉適配者類的一個(gè)或多個(gè)方法,就只好先做一個(gè)適配者類的子類,將適配者類的方法置換掉,然后再把適配者類的子類當(dāng)做真正的適配者進(jìn)行適配,實(shí)現(xiàn)過程較為復(fù)雜。
適用環(huán)境
在以下情況下可以使用適配器模式:
系統(tǒng)需要使用現(xiàn)有的類,而這些類的接口不符合系統(tǒng)的需要。
想要建立一個(gè)可以重復(fù)使用的類,用于與一些彼此之間沒有太大關(guān)聯(lián)的一些類,包括一些可能在將來引進(jìn)的類一起工作。
模式應(yīng)用
Sun公司在1996年公開了Java語言的數(shù)據(jù)庫連接工具JDBC,JDBC使得Java語言程序能夠與數(shù)據(jù)庫連接,并使用SQL語言來查詢和操作數(shù)據(jù)。JDBC給出一個(gè)客戶端通用的抽象接口,每一個(gè)具體數(shù)據(jù)庫引擎(如SQL Server、Oracle、MySQL等)的JDBC驅(qū)動(dòng)軟件都是一個(gè)介于JDBC接口和數(shù)據(jù)庫引擎接口之間的適配器軟件。抽象的JDBC接口和各個(gè)數(shù)據(jù)庫引擎API之間都需要相應(yīng)的適配器軟件,這就是為各個(gè)不同數(shù)據(jù)庫引擎準(zhǔn)備的驅(qū)動(dòng)程序。
模式擴(kuò)展
默認(rèn)適配器模式(Default Adapter Pattern)或缺省適配器模式
當(dāng)不需要全部實(shí)現(xiàn)接口提供的方法時(shí),可先設(shè)計(jì)一個(gè)抽象類實(shí)現(xiàn)接口,并為該接口中每個(gè)方法提供一個(gè)默認(rèn)實(shí)現(xiàn)(空方法),那么該抽象類的子類可有選擇地覆蓋父類的某些方法來實(shí)現(xiàn)需求,它適用于一個(gè)接口不想使用其所有的方法的情況。因此也稱為單接口適配器模式。
總結(jié)
結(jié)構(gòu)型模式描述如何將類或者對(duì)象結(jié)合在一起形成更大的結(jié)構(gòu)。
適配器模式用于將一個(gè)接口轉(zhuǎn)換成客戶希望的另一個(gè)接口,適配器模式使接口不兼容的那些類可以一起工作,其別名為包裝器。適配器模式既可以作為類結(jié)構(gòu)型模式,也可以作為對(duì)象結(jié)構(gòu)型模式。
適配器模式包含四個(gè)角色:目標(biāo)抽象類定義客戶要用的特定領(lǐng)域的接口;適配器類可以調(diào)用另一個(gè)接口,作為一個(gè)轉(zhuǎn)換器,對(duì)適配者和抽象目標(biāo)類進(jìn)行適配,它是適配器模式的核心;適配者類是被適配的角色,它定義了一個(gè)已經(jīng)存在的接口,這個(gè)接口需要適配;在客戶類中針對(duì)目標(biāo)抽象類進(jìn)行編程,調(diào)用在目標(biāo)抽象類中定義的業(yè)務(wù)方法。
在類適配器模式中,適配器類實(shí)現(xiàn)了目標(biāo)抽象類接口并繼承了適配者類,并在目標(biāo)抽象類的實(shí)現(xiàn)方法中調(diào)用所繼承的適配者類的方法;在對(duì)象適配器模式中,適配器類繼承了目標(biāo)抽象類并定義了一個(gè)適配者類的對(duì)象實(shí)例,在所繼承的目標(biāo)抽象類方法中調(diào)用適配者類的相應(yīng)業(yè)務(wù)方法。
適配器模式的主要優(yōu)點(diǎn)是將目標(biāo)類和適配者類解耦,增加了類的透明性和復(fù)用性,同時(shí)系統(tǒng)的靈活性和擴(kuò)展性都非常好,更換適配器或者增加新的適配器都非常方便,符合“開閉原則”;類適配器模式的缺點(diǎn)是適配器類在很多編程語言中不能同時(shí)適配多個(gè)適配者類,對(duì)象適配器模式的缺點(diǎn)是很難置換適配者類的方法。
適配器模式適用情況包括:系統(tǒng)需要使用現(xiàn)有的類,而這些類的接口不符合系統(tǒng)的需要;想要建立一個(gè)可以重復(fù)使用的類,用于與一些彼此之間沒有太大關(guān)聯(lián)的一些類一起工作