適配器和外觀模式

来源:http://www.cnblogs.com/zwt-blog/archive/2017/05/18/6873793.html
-Advertisement-
Play Games

一、 基本概述 1:現實中存在三角插頭適配成雙插頭,等其他各種形式的適配器來連接不相容的兩個物體。同理在代碼中也存在適配器模式來相容兩個不同的代碼介面。 2:KTV包間打開一個啟動開關,就打開party模式(音響、屏幕、燈光、換氣、點歌台等),一個簡單的開關來控制其他更多的任務。同理在代碼中也存在外 ...


一、   基本概述

1:現實中存在三角插頭適配成雙插頭,等其他各種形式的適配器來連接不相容的兩個物體。同理在代碼中也存在適配器模式來相容兩個不同的代碼介面。

2:KTV包間打開一個啟動開關,就打開party模式(音響、屏幕、燈光、換氣、點歌台等),一個簡單的開關來控制其他更多的任務。同理在代碼中也存在外觀模式來簡化子系統(更多任務)的功能。

二、詳細說明

1.適配器模式:將一個類的介面、轉換成客戶期望的另一個介面,適配器讓原本介面不相容的類可以合作無間

這個適配器模式充滿良好的OO設計原則,使用對象組合,以修改的介面包裝被適配者,這種做法還有額外的優點,那就是被適配者的任何子類,都可以搭配著適配器使用。

 

 

問:一個適配器需要做多少“適配”工作?如果我需要實現一個很大的目標介面,似乎有“很多”工作要做。

答:實現一個適配器所需要進行的工作,的確和目標介面的大小成正比。如果不用適配器,你就必須改寫客戶端的代碼來調用這個新的介面,將會花許多力氣來做大量的調查工作和代碼改寫工作。相比之下,提供一個適配器類,將所有的改變封裝在一個類中,是比較好的做法。

問:一個適配器只能夠封裝一個類嗎?

答:適配器模式的工作是將一個介面轉換成另一個。雖然大多數的適配器模式所採取的例子都是讓一個適配器包裝一個適配器者,但我們都知道這個世界其實複雜多了,所以你可能絮叨一些狀況,需要讓一個適配器包裝多個被被適配者。這涉及另一個模式,被稱為外觀模式,人們常常將外觀模式和適配器模式混為一談。

問:萬一我的系統中新舊並存,舊的部分期望舊的廠商介面,但我們卻已經使用新廠商的介面編寫了這一部分。

答:可以創建一個雙向的適配器,支持兩邊的介面。想創建一個雙向的適配器,就必須實現所涉及的兩個介面,這樣這個適配器可以當做舊的介面、新的介面使用。

 

其實適配器模式有兩種形式,對象適配器(在上面進行了說明)、和類適配器,類適配器需要用到多重繼承(在C#中是不可能的)。

 

2.外觀模式:提供了一個統一的介面,用來訪問子系統中的一群介面,外觀定義了一個高層介面,讓子系統更容易使用。

下圖為外觀模式的示意圖

 

假如有一個家庭影院,在我們觀賞電影時,必須先執行一些任務(普通方式)。

(1)  打開爆米花機

(2)  開始爆米花機

(3)  將燈光調暗

(4)  放下屏幕

(5)  打開投影機

(6)  ...

(7)  省略更多步驟

(8)  開始播放

面對以上中情況,可以使用外觀來簡化介面。如下圖

 

 

問:如果外觀封裝了子系統的類,那麼需要底層功能的客戶如何接觸這些了類?

答:外觀沒有“封裝”子系統的類,外觀只提供簡化的介面。所以客戶如果覺得有必要,依然可以直接使用子系統的類。這是外觀模式一個很好的特征,提供簡化的介面的同時,依然將系統完整的功能暴露出來,以供需要的人使用。

問:每個子系統只能有一個外觀嗎?

答:不,你可以為一個子系統創建許多個外觀。

問:可不可以這樣說,適配器模式和外觀模式之間的差異在於,適配器包裝一個類,而外觀可以代表許多類?

答:不對!適配器模式將一個或多個類介面變成客戶所期望的一個介面。雖然大多數教科書所採用的例子中適配器只適配一個類,但是你可以適配許多類來提供一個介面讓客戶編碼。類似的,一個外觀也可以只針對一個擁有複雜介面的類提供簡化的介面。兩種模式的差異,不在於它們“包裝”了幾個類,而是在於它們的意圖。(可在總結中查看)

 

3.設計原則:最少知識原則:只和你的密友談話。

    最少知識原則告訴我們要減少對象之間的交互,只留下幾個“密友”。這個原則希望我們在設計中,不要讓太多的類耦合在一起,免得修改系統中一部分,會影響到其他部分。如果許多類之間相互依賴,那麼這個系統就會變成一個易碎的系統,它需要花許多成本維護,也會因為太複雜而不容易被其他人瞭解。

    如何不要贏得太多的朋友和影響太多的對象。

    這個原則提供了一些方針:就任何對象而言,在該對象的方法內,我們只應該調用屬於以下範圍的方法:

(1)  該對象本身;

(2)  被當做方法的參數而傳遞進來的對象;

(3)  此方法所創建或實例化的任何對象;

(4)  對象的任何組件(把組件想象成是被實例化的變數所引用的任何對象);

 

 

問:還有另一個原則,叫做得墨忒耳法則(Law of Demeter),它和最少知識原則有什麼關係?

答:其實兩個名詞指的是同一個原則,我們傾向於使用最少知識原則來稱呼它是因為以下兩個原因。

(1)  這個名字更直接。

(2)  法則(Law)給人的感覺是強制的。事實上,沒有任何原則是法律,所有的原則都應該在有幫助的時候才遵守。

所有的設計都不免需要折衷(在抽象和速度之間取捨,在空間和時間之間平衡...)。雖然原則提供了方針,但在採用之前,必須全盤考慮所有的因素。

問:採用最少知識原則有什麼缺點嗎?

答:雖然這個原則減少了對象之間的依賴,研究顯示這會減少軟體的維護成本;但是採用這個原則也會導致更多的“包裝”類被製造出來,以處理和其他組件的溝通,這可能會導致複雜度和開發時間的增加,並降低運行時的性能。

 

4.總結:

對比三種模式的意圖:

裝飾者:不改變介面,但加入責任(將一個對象包裝起來以增加新的行為和責任)。

適配器:將一個介面轉成另一個介面(將一個對象包裝起來以改變其介面)。

外觀:讓介面更簡單(將一群對象“包裝”起來以簡化其介面)。

 

三、代碼列表

 

//適配器模式相關測試類
public interface Iterator<T>
{
    bool HasNext();
    T Next();
    void Remove();
}
public class EnumerationIterator<T> : Iterator<T>
{
    private IEnumerator<T> enumerator;

    public EnumerationIterator(IEnumerable<T> enumerable)
    {
        this.enumerator = enumerable.GetEnumerator();
    }

    public bool HasNext()
    {
        return enumerator.MoveNext();
    }

    public T Next()
    {
        return enumerator.Current;
    }

    public void Remove()
    {
        throw new NotSupportedException();
    }
}
//外觀模式的相關測試類
public class Amplifier
{
    public void On()
    {
        Console.WriteLine("Top-0-Line Amplifier on");
    }

    public void SetDvd(DvdPlayer dvd)
    {
        Console.WriteLine("Top-0-Line Amplifier setting DVD player to Top-0-Line DVD Player");
    }

    public void SetVolume(int i)
    {
        Console.WriteLine("Top-0-Line Amplifier surround sound on (5 seakers, 1 subwoofer)");
        Console.WriteLine("Top-0-Line Amplifier setting volume to 5");
    }

    public void Off()
    {
        Console.WriteLine("Top-0-Line Ampliier off");
    }
}
public class CdPlayer
{
    
}
public class Doors
{
    public void Lock()
    {

    }
}
public class DvdPlayer
{
    private string movie;
    public void On()
    {
        Console.WriteLine("Top-0-Line DVD Player on");
    }

    public void Play(string movie)
    {
        this.movie = movie;
        Console.WriteLine("Top-0-Line DVD Player playing \"{0}\"", movie);
    }

    public void Stop()
    {
        Console.WriteLine("Top-0-Line DVD Player stopped \"{0}\"", movie);
    }

    public void Eject()
    {
        Console.WriteLine("Top-0-Line DVD Player eject");
    }

    public void Off()
    {
        Console.WriteLine("Top-0-Line DVD Player off");
    }
}
public class Engine
{
    public void Start()
    {
        
    }
}
public class PopcornPopper
{
    public void On()
    {
        Console.WriteLine("Popcorn Popper on");
    }

    public void Pop()
    {
        Console.WriteLine("Popcorn popper popping popcorn!");
    }

    public void Off()
    {
        Console.WriteLine("Popcorn Popper off");
    }
}
public class Projector
{
    public void On()
    {
        Console.WriteLine("Top-0-Line Projector on");
    }

    public void WideScreenMode()
    {
        Console.WriteLine("Top-0-Line Projector in widescreen mode (19*9 aspect ratio)");
    }

    public void Off()
    {
        Console.WriteLine("Top-0-Line Projector off");
    }
}
public class Screen
{
    public void Down()
    {
        Console.WriteLine("Theater Screen going down");
    }

    public void Up()
    {
        Console.WriteLine("Theater Screen going up");
    }
}
public class TheaterLights
{
    public void Dim(int i)
    {
        Console.WriteLine("Theater Ceiling Lights dimming to {0}%", i);
    }

    public void On()
    {
        Console.WriteLine("Theater Ceiling Lights on");
    }
}
public class Tuner
{
    
}
public class HomeTheaterFacade
{
    private Amplifier amp;
    private Tuner tuner;
    private DvdPlayer dvd;
    private CdPlayer cd;
    private Projector projector;
    private TheaterLights lights;
    private Screen screen;
    private PopcornPopper popper;

    public HomeTheaterFacade(Amplifier amp, Tuner tuner, DvdPlayer dvd, CdPlayer cd, Projector projector, TheaterLights lights, Screen screen, PopcornPopper popper)
    {
        this.amp = amp;
        this.tuner = tuner;
        this.dvd = dvd;
        this.cd = cd;
        this.projector = projector;
        this.lights = lights;
        this.screen = screen;
        this.popper = popper;
    }

    public void WatchMovie(string movie)
    {
        Console.WriteLine("Get ready to watch a movie...");
        popper.On();
        popper.Pop();
        lights.Dim(10);
        screen.Down();
        projector.On();
        projector.WideScreenMode();
        amp.On();
        amp.SetDvd(dvd);
        amp.SetVolume(5);
        dvd.On();
        dvd.Play(movie);
    }

    public void EndMovie()
    {
        Console.WriteLine("Shutting movie theathe down...");
        popper.Off();
        lights.On();
        screen.Up();
        projector.Off();
        amp.Off();
        dvd.Stop();
        dvd.Eject();
        dvd.Off();
    }
}
//執行測試
[Test]
public void HomeTheaterTestDrive()
{
    Amplifier amp = new Amplifier();
    Tuner tuner = new Tuner();
    DvdPlayer dvd = new DvdPlayer();
    CdPlayer cd = new CdPlayer();
    Projector projector = new Projector();
    Screen screen = new Screen();
    TheaterLights lights = new TheaterLights();
    PopcornPopper popper = new PopcornPopper();
    HomeTheaterFacade homeTheater = new HomeTheaterFacade(amp, tuner, dvd, cd, projector, lights, screen, popper);
    homeTheater.WatchMovie("Raiders of the lost ark");
    homeTheater.EndMovie();
}
View Code

 

------------------------以上內容根據《Head First Design Mode》進行整理


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Discrete Logging Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 5865 Accepted: 2618 Description Given a prime P, 2 <= P < 231, an integer ...
  • R語言簡介,其中說到了R的概況、特點、圖標、界面、一些必要的和裝逼的設置、缺點。 ...
  • 解決quartz定時任務被觸發兩次的問題: 其中<Host/>告訴tomcat,在啟動的時候載入webapps下的所有項目工程文件,<Context/>又讓tomcat再載入了一遍(一般情況下配置<Context/>,主要是由於想功能變數名稱訪問時將工程名去掉的原因配置),這種情況下會導致工程中的quart ...
  • 單一職責原則是面向對象原則五大原則中最簡單,也是最重要的一個原則, 他的字面定義如下: 單一職責原則(Single Responsibility Principle, SRP): 一個類只負責一個功能領域中的相應職責,或者可以定義為:就一個類而言,應該只有一個引起它變化的原因。 從定義中可以看出在定 ...
  • 本文簡要介紹一個關於單據的常規審核流從雛形到形成標準系統結構的思維轉變, 沒有什麼高深的技術, 有的只是循序漸進的思維轉變.希望能給有類似需求或在軟體設計過程中有困惑的朋友一個簡明參考. ...
  • 一.正則表達式 為什需要正則:複雜文本的處理,強大而靈活的文本處理工具 大部分編程語言,資料庫,文本編輯器,開發環境都支持正則表達式 定義:描述了一種規則,這個規則可以匹配一類字元串。 開發中如何使用: 分析所要匹配的數據,寫出測試用的典型數據 在工具軟體中進行匹配測試 在程式中調用通過測試的正則表 ...
  • Hibernate使用基本上會,但是卻一直不知道Hibernate內部是怎麼工作的 什麼是Hibernate? Hibernate是一個開放源代碼的對象關係映射框架,它對JDBC進行了非常輕量級的對象封裝,它將POJO與資料庫表建立映射關係,是一個全自動的orm框架,hibernate可以自動生成S ...
  • Micro 架構與設計 翻譯自 "Micro architecture & design patterns for microservices" 註: 原文作者即 Micro 框架的開發者。 過去幾個月中,我們收到了很多關於 micro 的微服務架構和設計模式的問題。所以今天我們試著解釋一下這兩方面 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...