設計模式(八)橋接模式(Bridge Pattern)

来源:http://www.cnblogs.com/xiongzaiqiren/archive/2016/03/25/5320352.html
-Advertisement-
Play Games

一、引言 這裡以電視遙控器的一個例子來引出橋接模式解決的問題,首先,我們每個牌子的電視機都有一個遙控器,此時我們能想到的一個設計是——把遙控器做為一個抽象類,抽象類中提供遙控器的所有實現,其他具體電視品牌的遙控器都繼承這個抽象類,具體設計類圖如下: 這樣的實現使得每部不同型號的電視都有自己遙控器實現 ...


一、引言

  這裡以電視遙控器的一個例子來引出橋接模式解決的問題,首先,我們每個牌子的電視機都有一個遙控器,此時我們能想到的一個設計是——把遙控器做為一個抽象類,抽象類中提供遙控器的所有實現,其他具體電視品牌的遙控器都繼承這個抽象類,具體設計類圖如下:

  這樣的實現使得每部不同型號的電視都有自己遙控器實現,這樣的設計對於電視機的改變可以很好地應對,只需要添加一個派生類就搞定了,但隨著時間的推移,用戶需要改變遙控器的功能,如:用戶可能後面需要對遙控器添加返回上一個台等功能時,此時上面的設計就需要修改抽象類RemoteControl的提供的介面了,此時可能只需要向抽象類中添加一個方法就可以解決了,但是這樣帶來的問題是我們改變了抽象的實現,如果用戶需要同時改變電視機品型號和遙控器功能時,上面的設計就會導致相當大的修改,顯然這樣的設計並不是好的設計。然而使用橋接模式可以很好地解決這個問題,下麵讓我具體看看橋接模式是如何實現的。

二、橋接模式的詳細介紹

2.1 定義

  橋接模式即將抽象部分與實現部分脫耦,使它們可以獨立變化。對於上面的問題中,抽象化也就是RemoteControl類,實現部分也就是On()、Off()、NextChannel()等這樣的方法(即遙控器的實現),上面的設計中,抽象化和實現部分在一起,橋接模式的目的就是使兩者分離,根據面向對象的封裝變化的原則,我們可以把實現部分的變化(也就是遙控器功能的變化)封裝到另外一個類中,這樣的一個思路也就是橋接模式的實現,大家可以對照橋接模式的實現代碼來解決我們的分析思路。

2.2 橋接模式實現

上面定義部分已經給出了我們橋接模式的目的以及實現思路了,下麵讓我們具體看看橋接模式是如何解決引言部分設計的不足。

抽象化部分的代碼:

    /// <summary>
    /// 抽象概念中的遙控器,扮演抽象化角色
    /// </summary>
    public class RemoteControl
    {
        // 欄位
        private TV implementor;
        // 屬性
        public TV Implementor
        {
            get { return implementor; }
            set { implementor = value; }
        }
        /// <summary>
        /// 開電視機,這裡抽象類中不再提供實現了,而是調用實現類中的實現
        /// </summary>
        public virtual void On()
        {
            implementor.On();
        }
        /// <summary>
        /// 關電視機
        /// </summary>
        public virtual void Off()
        {
            implementor.Off();
        }
        /// <summary>
        /// 換頻道
        /// </summary>
        public virtual void SetChannel()
        {
            implementor.tuneChannel();
        }
    }
    /// <summary>
    /// 具體遙控器
    /// </summary>
    public class ConcreteRemote : RemoteControl
    {
        public override void SetChannel()
        {
            Console.WriteLine("---------------------");
            base.SetChannel();
            Console.WriteLine("---------------------");
        }
    }

遙控器的實現方法部分代碼,即實現化部分代碼,此時我們用另外一個抽象類TV封裝了遙控器功能的變化,具體實現交給具體型號電視機去完成:

    /// <summary>
    /// 電視機,提供抽象方法
    /// </summary>
    public abstract class TV
    {
        public abstract void On();
        public abstract void Off();
        public abstract void tuneChannel();
    }
    /// <summary>
    /// 長虹牌電視機,重寫基類的抽象方法
    /// 提供具體的實現
    /// </summary>
    public class ChangHong : TV
    {
        public override void On()
        {
            Console.WriteLine("長虹牌電視機已經打開了");
        }
        public override void Off()
        {
            Console.WriteLine("長虹牌電視機已經關掉了");
        }
        public override void tuneChannel()
        {
            Console.WriteLine("長虹牌電視機換頻道");
        }
    }
    /// <summary>
    /// 三星牌電視機,重寫基類的抽象方法
    /// </summary>
    public class Samsung : TV
    {
        public override void On()
        {
            Console.WriteLine("三星牌電視機已經打開了");
        }
        public override void Off()
        {
            Console.WriteLine("三星牌電視機已經關掉了");
        }
        public override void tuneChannel()
        {
            Console.WriteLine("三星牌電視機換頻道");
        }
    }

採用橋接模式的客戶端調用代碼:

    /// <summary>
    /// 以電視機遙控器的例子來演示橋接模式
    /// </summary>
    class Client
    {
        static void Main(string[] args)
        {
            // 創建一個遙控器
            RemoteControl remoteControl = new ConcreteRemote();
            // 長虹電視機
            remoteControl.Implementor = new ChangHong();
            remoteControl.On();
            remoteControl.SetChannel();
            remoteControl.Off();
            Console.WriteLine();
            // 三星牌電視機
            remoteControl.Implementor = new Samsung();
            remoteControl.On();
            remoteControl.SetChannel();
            remoteControl.Off();
            Console.Read();
        }
    }

  上面橋接模式的實現中,遙控器的功能實現方法不在遙控器抽象類中去實現了,而是把實現部分用來另一個電視機類去封裝它,然而遙控器中只包含電視機類的一個引用,同時這樣的設計也非常符合現實生活中的情況(我認為的現實生活中遙控器的實現——遙控器中並不包含換台,打開電視機這樣的功能的實現,遙控器只是包含了電視機上這些功能的引用,然後紅外線去找到電視機上對應功能的的實現)。通過橋接模式,我們把抽象化和實現化部分分離開了,這樣就可以很好應對這兩方面的變化了。

2.3 橋接模式的類圖

看完橋接模式的實現後,為了幫助大家理清對橋接模式中類之間關係,這裡給出橋接模式的類圖結構:

三、橋接模式的優缺點

介紹完橋接模式,讓我們看看橋接模式具體哪些優缺點。

優點:

  把抽象介面與其實現解耦。

  抽象和實現可以獨立擴展,不會影響到對方。

  實現細節對客戶透明,對用於隱藏了具體實現細節。

缺點:

  增加了系統的複雜度

四、使用場景

我們再來看看橋接模式的使用場景,在以下情況下應當使用橋接模式:

  1,如果一個系統需要在構件的抽象化角色和具體化角色之間添加更多的靈活性,避免在兩個層次之間建立靜態的聯繫。
  2,設計要求實現化角色的任何改變不應當影響客戶端,或者實現化角色的改變對客戶端是完全透明的。
  3,需要跨越多個平臺的圖形和視窗系統上。
  4,一個類存在兩個獨立變化的維度,且兩個維度都需要進行擴展。

五、一個實際應用橋接模式的例子

橋接模式也經常用於具體的系統開發中,對於三層架構中就應用了橋接模式,三層架構中的業務邏輯層BLL中通過橋接模式與數據操作層解耦(DAL),其實現方式就是在BLL層中引用了DAL層中一個引用。這樣數據操作的實現可以在不改變客戶端代碼的情況下動態進行更換,下麵看一個簡單的示例代碼:

    // 客戶端調用
    // 類似Web應用程式
    class Client
    {
        static void Main(string[] args)
        {
            BusinessObject customers = new CustomersBusinessObject("ShangHai");
            customers.Dataacces = new CustomersDataAccess();
            customers.Add("小六");
            Console.WriteLine("增加了一位成員的結果:");
            customers.ShowAll();
            customers.Delete("王五");
            Console.WriteLine("刪除了一位成員的結果:");
            customers.ShowAll();
            Console.WriteLine("更新了一位成員的結果:");
            customers.Update("Learning_Hard");
            customers.ShowAll();
            Console.Read();
        }
    }
    // BLL 層
    public class BusinessObject
    {
        // 欄位
        private DataAccess dataacess;
        private string city;
        public BusinessObject(string city)
        {
            this.city = city;
        }
        // 屬性
        public DataAccess Dataacces
        {
            get { return dataacess; }
            set { dataacess = value; }
        }
        // 方法
        public virtual void Add(string name)
        {
            Dataacces.AddRecord(name);
        }
        public virtual void Delete(string name)
        {
            Dataacces.DeleteRecord(name);
        }
        public virtual void Update(string name)
        {
            Dataacces.UpdateRecord(name);
        }
        public virtual string Get(int index)
        {
            return Dataacces.GetRecord(index);
        }
        public virtual void ShowAll()
        {
            Console.WriteLine();
            Console.WriteLine("{0}的顧客有:", city);
            Dataacces.ShowAllRecords();
        }
    }
    public class CustomersBusinessObject : BusinessObject
    {
        public CustomersBusinessObject(string city)
            : base(city) { }
        // 重寫方法
        public override void ShowAll()
        {
            Console.WriteLine("------------------------");
            base.ShowAll();
            Console.WriteLine("------------------------");
        }
    }
    /// <summary>
    /// 相當於三層架構中數據訪問層(DAL)
    /// </summary>
    public abstract class DataAccess
    {
        // 對記錄的增刪改查操作
        public abstract void AddRecord(string name);
        public abstract void DeleteRecord(string name);
        public abstract void UpdateRecord(string name);
        public abstract string GetRecord(int index);
        public abstract void ShowAllRecords();
    }
    public class CustomersDataAccess:DataAccess
    {
        // 欄位
        private List<string> customers =new List<string>();
        public CustomersDataAccess()
        {
            // 實際業務中從資料庫中讀取數據再填充列表
            customers.Add("Learning Hard");
            customers.Add("張三");
            customers.Add("李四");
            customers.Add("王五");
        }
        // 重寫方法
        public override void AddRecord(string name)
        {
            customers.Add(name);
        }
        public override void DeleteRecord(string name)
        {
            customers.Remove(name);
        }
        public override void UpdateRecord(string updatename)
        {
            customers[0] = updatename;
        }
        public override string GetRecord(int index)
        {
            return customers[index];
        }
        public override void ShowAllRecords()
        {
            foreach (string name in customers)
            {
                Console.WriteLine(" " + name);
            }
        }
             
    }

六、總結

到這裡,橋接模式就介紹完了,橋接模式實現了抽象化與實現化的解耦,使它們相互獨立互不影響到對方

 

 

以上內容摘抄自:http://learninghard.blog.51cto.com/6146675/1310140

 


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

-Advertisement-
Play Games
更多相關文章
  • 1. 創建一個context processor函數 新建一個文件命名為custom_processors.py,把它放到項目app文件夾(例如我的blog文件夾),添加一個返回字典的函數,其代碼如下: from sets import Set from django.db.models impor ...
  • 從今天起, 我會每天把閱讀tiny_cnn的閱讀心得提交到博客園中希望大家在這個平臺上可以多多交流; 關於如果閱讀代碼? 抓住重點,忽略細節 首先打開從github上下載的文件: 通過csdn和網上搜索一番會知道這個文件的各個目錄存放的是什麼; 我用${root} 代表到tiny-cnn-maste... ...
  • 命名空間提供了一種從邏輯上組織類的方式,防止命名衝突。 幾種常見語言 C++ 命名空間是可以嵌套的 嵌套的命名空間是指定義在其他命名空間中的命名空間。嵌套的命名空間是一個嵌套的作用域,內層命名空間聲明的名字將隱藏外層命名空間聲明的同名成員: C++ int x = 20; namespace out ...
  • 在前面幾次討論中我們介紹了Free是個產生Monad的最基本結構。它的原理是把一段程式(AST)一連串的運算指令(ADT)轉化成數據結構存放在記憶體里,這個過程是個獨立的功能描述過程。然後另一個獨立運算過程的Interpreter會遍歷(traverse)AST結構,讀取結構里的運算指令,實際運行指令 ...
  • 1 #-*- coding:utf-8 -*- 2 3 import curses 4 from random import randrange, choice # generate and place new tile 5 from collections import defaultdict 6 ...
  • 題目: 計算字元串最後一個單詞的長度,單詞以空格隔開。 輸入: 一行字元串。 輸出: 整數N,最後一個單詞的長度。 例如: hello world 輸出: 5 思路: 考慮到最後一個單詞中可能含有空格,例如: “Hello world#" (#表示空格) int lenOfLastWord(cons... ...
  • 用PHPExcel做導出execl的時候發現在本地沒有問題,但是把網站傳到租用的伺服器的時候就報錯,具體如下: 代碼如下: ...
  • 在CRM(客戶關係管理)系統或者其他業務支撐型系統的開發過程中,最經常多變的就是複雜的業務規則。因為這些規則要迎合、順應市場的變化,如何能有效到做到業務規則和整體的系統支撐架構解耦分離,這個是開發過程中必須考慮的一個問題。每當客戶要求改變一個業務規則的時候,我們又如何能做到在最短的時間內完成需求的開 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...