這裏以電視遙控器的一個例子來引出橋接模式解決的問題,首先,咱們每一個牌子的電視機都有一個遙控器,此時咱們能想到的一個設計是——把遙控器作爲一個抽象類,抽象類中提供遙控器的全部實現,其餘具體電視品牌的遙控器都繼承這個抽象類,具體設計類圖以下:數據庫
這樣的實現使得每部不一樣型號的電視都有本身遙控器實現,這樣的設計對於電視機的改變能夠很好地應對,只須要添加一個派生類就搞定了,但隨着時間的推移,用戶須要改變遙控器的功能,如:用戶可能後面須要對遙控器添加返回上一個臺等功能時,此時上面的設計就須要修改抽象類RemoteControl的提供的接口了,此時可能只須要向抽象類中添加一個方法就能夠解決了,可是這樣帶來的問題是咱們改變了抽象的實現,若是用戶須要同時改變電視機品型號和遙控器功能時,上面的設計就會致使至關大的修改,顯然這樣的設計並非好的設計。然而使用橋接模式能夠很好地解決這個問題,下面讓我具體看看橋接模式是如何實現的。c#
橋接模式即將抽象部分與實現部分脫耦,使它們能夠獨立變化。對於上面的問題中,抽象化也就是RemoteControl類,實現部分也就是On()、Off()、NextChannel()等這樣的方法(即遙控器的實現),上面的設計中,抽象化和實現部分在一塊兒,橋接模式的目的就是使二者分離,根據面向對象的封裝變化的原則,咱們能夠把實現部分的變化(也就是遙控器功能的變化)封裝到另一個類中,這樣的一個思路也就是橋接模式的實現,你們能夠對照橋接模式的實現代碼來解決咱們的分析思路。架構
上面定義部分已經給出了咱們橋接模式的目的以及實現思路了,下面讓咱們具體看看橋接模式是如何解決引言部分設計的不足。ide
抽象化部分的代碼:this
/// <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封裝了遙控器功能的變化,具體實現交給具體型號電視機去完成:spa
/// <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(); } }
上面橋接模式的實現中,遙控器的功能實現方法不在遙控器抽象類中去實現了,而是把實現部分用來另外一個電視機類去封裝它,然而遙控器中只包含電視機類的一個引用,同時這樣的設計也很是符合現實生活中的狀況(我認爲的現實生活中遙控器的實現——遙控器中並不包含換臺,打開電視機這樣的功能的實現,遙控器只是包含了電視機上這些功能的引用,而後紅外線去找到電視機上對應功能的的實現)。經過橋接模式,咱們把抽象化和實現化部分分離開了,這樣就能夠很好應對這兩方面的變化了。對象
看完橋接模式的實現後,爲了幫助你們理清對橋接模式中類之間關係,這裏給出橋接模式的類圖結構:繼承
介紹完橋接模式,讓咱們看看橋接模式具體哪些優缺點。接口
優勢:
把抽象接口與其實現解耦。
抽象和實現能夠獨立擴展,不會影響到對方。
實現細節對客戶透明,對用於隱藏了具體實現細節。
缺點: 增長了系統的複雜度
咱們再來看看橋接模式的使用場景,在如下狀況下應當使用橋接模式:
若是一個系統須要在構件的抽象化角色和具體化角色之間添加更多的靈活性,避免在兩個層次之間創建靜態的聯繫。
設計要求實現化角色的任何改變不該當影響客戶端,或者實現化角色的改變對客戶端是徹底透明的。
須要跨越多個平臺的圖形和窗口系統上。
一個類存在兩個獨立變化的維度,且兩個維度都須要進行擴展。
橋接模式也常常用於具體的系統開發中,對於三層架構中就應用了橋接模式,三層架構中的業務邏輯層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); } } }
到這裏,橋接模式的介紹就介紹,橋接模式實現了抽象化與實現化的解耦,使它們相互獨立互不影響到對方。