「補課」進行時:設計模式(12)——適配器模式

1. 前文匯總

「補課」進行時:設計模式系列

2. 適配器模式

2.1 定義

適配器模式(Adapter Pattern)的定義如下:

Convert the interface of a class into another interface clients expect.Adapterlets classes work together that couldn’t otherwise because of incompatibleinterfaces.(將一個類的介面變換成客戶端所期待的另一種介面,從而使原本因介面不匹配而無法在一起工作的兩個類能夠在一起工作。)

  • Target: 目標角色,該角色定義把其他類轉換為何種介面,也就是我們的期望介面。
  • Adaptee: 源角色,它是已經存在的、運行良好的類或對象,經過適配器角色的包裝,它會成為一個新的角色。
  • Adapter: 適配器角色,核心角色,其他兩個角色都是已經存在的角色,而適配器角色是需要新建立的,它的職責非常簡單:通過繼承或是類關聯的方式把源角色轉換為目標角色。

目標角色:

public interface Target {
    // 目標自己的方法
    void request();
}

具體目標角色實現:

public class ConcreteTarget implements Target{
    @Override
    public void request() {
        System.out.println("Target method is run!");
    }
}

源角色:

public class Adaptee {
    public void doSomething(){
        System.out.println("Adaptee method doSomething is run!");
    }
}

適配器:

public class Adapter extends Adaptee implements Target {
    @Override
    public void request() {
        super.doSomething();
    }
}

測試類:

public class Test {
    public static void main(String[] args) {
        // 原有業務邏輯
        Target target = new ConcreteTarget();
        target.request();
        // 經過適配器裝飾後的業務邏輯
        Target target1 = new Adapter();
        target1.request();
    }
}

2.2 優點

  • 適配器模式可以讓兩個沒有任何關係的類在一起運行。
  • 增加了類的透明性:我們訪問的Target目標角色,但是具體的實現都委託給了源角色,而這些對高層次模組是透明的,也是它不需要關心的。
  • 提高了類的復用度:源角色在原有的系統中還是可以正常使用,而在目標角色中也可以充當新的演員。

3. 案例

適配器模式最好的案例就是我們經常使用的電源,我們國家的民用電都是 220V 的交流電,而在日本是使用的 110V 的交流電,而我們的手機充電,只需要使用 5V 的就可以了,或者還有我們的筆記型電腦電腦,使用的是 19V 或者也有使用 24V 的。

首先,定義一個輸出交流電的介面,希望是可以輸出 110V 的交流電,因為中國的生產的插頭都是 220V 的,當我們購買了直接進口到中國的電器後,只有 110V 的交流電才能使用:

public interface AC {
    int convert110v();
}

上面順便帶上了這個交流電介面的兩個實現。

接著,創建一個源類,因為是我們中國的插頭,目前只能輸出 220V 的交流電

public class PowerPlug {
    public void OutPut220V() {
        System.out.println("現有插頭只能輸出 220V 電壓");
    }
}

接著,我們裝一個適配器,將 220V 的電壓轉換成 110V 的電壓,可以使我們進口回來的電器使用:

public class Adapter220 extends PowerPlug implements AC {
    @Override
    public void convert110v() {
        this.OutPut220V();
        System.out.println("通過適配器,將 220V 轉換成 110V");
    }
}

這裡雖然我們輸出的是 110V 的電壓,但是實際上還是調用的之前的 220V 的電壓,我們只是在適配器這一步將電壓轉化成了 110V 。

最後是一個測試類:

public class Test {
    public static void main(String[] args) {
        AC ac = new Adapter220();
        ac.convert110v();
    }
}

最後的執行結果:

現有插頭只能輸出 220V 電壓
通過適配器,將 220V 轉換成 110V

最後在測試類當中,我們雖然是 new 了一個 220V 的實例出來,但是在做方法調用的時候,直接調用了適配器的方法,就相當於原本的輸出還是 220V ,只是套了一個將 220V 轉成 110V 的外殼。