java設計模式-調停者模式(Mediator)

定義

調停者模式是對象的行為模式。調停者模式包裝了一系列對象相互作用的方式,使得這些對象不必相互明顯引用,從而使它們可以較松散的耦合。當這些對象中的某些對象之間的相互作用發生改變時,不會立即影響到其他的一些對象之間的相互引用,從而保證這些相互作用可以彼此獨立的變化。

為什么需要調停者

如下圖所示,這個示意圖中有大量的對象,這些對象既會影響別的對象,又會被別的對象所影響,因此常常叫做同事對象Colleague。這些同事對象通過彼此的相互作用形成系統的行為。從圖中可以看出,幾乎每一個對象都需要與其他的對象發生相互左右,而這種相互作用表現為一個對象與另一個對象的直接耦合。這就是過度耦合的系統。

通過引入調停者對象Mediator,可以將系統的網狀結構變成以中介為中心的星形結構,如下圖所示。在這個星形結構中,同事對象不再通過直接的聯系與另一個對象發生相互作用;相反的,它通過調停者對象與另一個對象發生相互作用。調停者對象的存在保證了對象結構上的穩定,也就是說,系統的結構不會因為新對象的引入造成大量的修改工作。

調停者對象示意圖

一個好的面向對象的設計可以使對象之間增加協作性Collaboration,減少耦合度Couping。一個深思熟慮的設計會把一個系統分解為一群相互協作的同事對象,然后給每一個同事對象以獨特的責任,恰當的配置它們之間的協作關系,使他們可以在一起工作。

如果沒有主板

大家都知道,電腦里面各個配件之間的交互,主要是通過主板來完成的。如果電腦里面沒有了主板,那么各個配件之間就必須自行相互交互,以互相傳送數據。而且由于各個配件的接口不通,相互之間交互時,還必須把數據接口進行轉換才能匹配上。

所幸是有了主板,各個配件之間的交互完全通過主板來完成,每個配件都只需要和主板交互,而主板知道如何跟所有的配件打交道,這樣就簡單多了。

調停者模式的結構

調停者模式的示意性類圖如下所示:

調停者模式的示意性類圖

調停者模式包括以下角色:

  • 抽象調停者角色(Mediator):定義出同事對象到調停者對象的接口,其中主要方法是一個(或多個)事件方法。
  • 具體調停者角色(ConcreteMediator):實現了抽象調停者所聲明的事件方法。具體調停者知曉所有的具體同事類,并負責具體的協調各同事對象的交互關系。
  • 抽象同事類角色(Colleague):定義出調停者到同事對象的接口。同事對象只知道調停者而不知道其余的同事對象。
  • 具體同事類角色(ConcreteColleague):所有的具體同事類均從抽象同事類繼承而來。實現自己的業務,在需要與其他同事通信的時候,就與持有的調停者通信,調停者會負責與其他的同事交互。

示例代碼

抽象調停者類

public interface Mediator {
    /**
     * 同事對象在自身改變的時候來通知調停者方法
     * 讓調停者去負責相應的與其他同事對象的交互
     * @param c
     */
    public void changed(Colleague c);
}

具體調停者類

public class ConcreteMediator implements Mediator {
    //持有并維護同事A
    private ConcreteColleagueA colleagueA;
    //持有并維護同事B
    private ConcreteColleagueB colleagueB;
    
    public void setColleagueA(ConcreteColleagueA colleagueA) {
        this.colleagueA = colleagueA;
    }

    public void setColleagueB(ConcreteColleagueB colleagueB) {
        this.colleagueB = colleagueB;
    }

    public void changed(Colleague c) {
        /**
         * 某一個同事類發生了變化,同行需要與其他同事交互
         * 具體協調相應的同事對象來實現協同行為
         */
    }
}

抽象同事類

public class Colleague {
    /**
     * 持有一個調停者對象
     */
    private Mediator mediator;
    /**
     * 構造函數
     * @param mediator
     */
    public Colleague(Mediator mediator) {
        this.mediator = mediator;
    }
    /**
     * 獲取當前同事類對應的調停者對象
     * @return
     */
    public Mediator getMediator() {
        return this.mediator;
    }
}

具體同事類

public class ConcreteColleagueA extends Colleague {
    public ConcreteColleagueA(Mediator mediator) {
        super(mediator);
    }
    
    /**
     * 示意方法,執行某些操作
     */
    private void operation() {
        //在需要跟其他同事通信的時候,通知調停者對象
        getMediator().changed(this);
    }
}
public class ConcreteColleagueB extends Colleague {
    public ConcreteColleagueB(Mediator mediator) {
        super(mediator);
    }
    
    /**
     * 示意方法,執行某些操作
     */
    private void operation() {
        //在需要跟其他同事通信的時候,通知調停者對象
        getMediator().changed(this);
    }
}

使用電腦來看電影

在日常生活中,我們經常使用電腦來看電影,把這個過程描述出來,簡化后假定會有如下的交互過程:

  1. 首先是光驅要讀取光盤上的數據,然后告訴數據,它的狀態改變了。
  2. 主板去得到光驅的數據,把這些數據交給CPU進行分析處理。
  3. CPU處理完后,把數據分成了視頻數據和音頻數據,通知主板,它處理完了。
  4. 主板去得到CPU處理過后的數據,分別把數據交給顯卡和聲卡,去顯示出視頻和發出聲音。

要使用調停者模式來實現示例,那就要區分出同事對象和調停者對象。很明顯,主板是調停者,而光驅、聲卡、CPU、顯卡等配件,都是作為同事對象。

示例代碼

抽象同事類

public abstract class Colleague {
    /**
     * 持有一個調停者對象
     */
    private Mediator mediator;
    /**
     * 構造函數
     * @param mediator
     */
    public Colleague(Mediator mediator) {
        this.mediator = mediator;
    }
    /**
     * 獲取當前同事類對應的調停者方法
     * @return
     */
    public Mediator getMediator() {
        return mediator;
    }
}

同事類——光驅

public class CDDriver extends Colleague {
    /**
     * 光驅讀取出來的數據
     */
    private String data = "";
    /**
     * 構造函數
     * @param mediator
     */
    public CDDriver(Mediator mediator) {
        super(mediator);
    }
    /**
     * 獲取光盤讀取出來的數據
     * @return
     */
    public String getData() {
        return data;
    }
    /**
     * 讀取光盤
     */
    public void readCD() {
        //逗號前是視頻顯示的數據,逗號后是聲音
        this.data = "One Piece, 海賊王我當定了";
        //通知主板,自己的狀態發生了改變
        getMediator().changed(this);
    }
}

同事類——CPU

public class CPU extends Colleague {
    /**
     * 分解出來的視頻數據
     */
    private String videoData = "";
    /**
     * 分解出來的聲音數據
     */
    private String soundData = "";

    /**
     * 構造函數
     * 
     * @param mediator
     */
    public CPU(Mediator mediator) {
        super(mediator);
        // TODO Auto-generated constructor stub
    }

    /**
     * 獲取分解出來的視頻數據
     * 
     * @return
     */
    public String getVideoData() {
        return videoData;
    }

    /**
     * 獲取分解出來的聲音數據
     * 
     * @return
     */
    public String getSoundData() {
        return soundData;
    }

    /**
     * 處理數據,把數據分成音頻和視頻的數據
     * 
     * @param data
     */
    public void executeData(String data) {
        // 把數據分解開,前面是視頻數據,后面是音頻數據
        String[] array = data.split(",");
        this.videoData = array[0];
        this.soundData = array[1];
        // 通知主板,CPU完成工作
        getMediator().changed(this);
    }
}

同事類——顯卡

public class VideoCard extends Colleague {
    /**
     * 構造函授
     * @param mediator
     */
    public VideoCard(Mediator mediator) {
        super(mediator);
        // TODO Auto-generated constructor stub
    }
    /**
     * 顯示視頻數據
     * @param data
     */
    public void showData(String data) {
        System.out.println("您正在觀看的是:" + data);
    }
}

同事類——聲卡

public class SoundCard extends Colleague {
    /**
     * 構造函數
     * 
     * @param mediator
     */
    public SoundCard(Mediator mediator) {
        super(mediator);
    }

    /**
     * 按照聲頻數據發出聲音
     * @param data
     */
    public void soundData(String data) {
        System.out.println("畫外音:" + data);
    }
}

抽象調停者類

public interface Mediator {
    /**
     * 同事對象在自身改變的時候來通知調停者方法
     * 讓調停者去負責相應的與其他同事對象的交互
     * @param c
     */
    public void changed(Colleague c);
}

具體調停者類

public class MainBoard implements Mediator {
    /**
     * 需要知道要交互的同事類 - 光驅類
     */
    private CDDriver cdDriver = null;
    /**
     * 需要知道要交互的同事類 - CPU類
     */
    private CPU cpu = null;
    /**
     * 需要知道要交互的同事類 - 顯卡類
     */
    private VideoCard videoCard = null;
    /**
     * 需要知道要交互的同事類 - 聲卡類
     */
    private SoundCard soundCard = null;

    public void setCdDriver(CDDriver cdDriver) {
        this.cdDriver = cdDriver;
    }

    public void setCpu(CPU cpu) {
        this.cpu = cpu;
    }

    public void setVideoCard(VideoCard videoCard) {
        this.videoCard = videoCard;
    }

    public void setSoundCard(SoundCard soundCard) {
        this.soundCard = soundCard;
    }

    public void changed(Colleague c) {
        if (c instanceof CDDriver) {
            // 表示光驅讀取數據了
            this.openCDDriverReadData((CDDriver) c);
        } else if (c instanceof CPU) {
            this.openCPU((CPU) c);
        }
    }

    /**
     * 處理光驅讀取數據以后與其他對象的交互
     * 
     * @param cd
     */
    private void openCDDriverReadData(CDDriver cd) {
        // 先獲取光驅讀取的數據
        String data = cd.getData();
        // 把這些數據傳遞給CPU進行處理
        cpu.executeData(data);
    }

    /**
     * 處理CPU處理完數據后與其他對象的交互
     * 
     * @param cpu
     */
    private void openCPU(CPU cpu) {
        // 先獲取CPU處理后的數據
        String videoData = cpu.getVideoData();
        String soundData = cpu.getSoundData();
        // 把這些數據傳遞給顯卡和聲卡展示出來
        videoCard.showData(videoData);
        soundCard.soundData(soundData);
    }
}

客戶端類

public class Client {
    public static void main(String[] args) {
        // 創建調停者 -- 主板
        MainBoard mediator = new MainBoard();
        // 創建同事類
        CDDriver cdDriver = new CDDriver(mediator);
        CPU cpu = new CPU(mediator);
        VideoCard videoCard = new VideoCard(mediator);
        SoundCard soundCard = new SoundCard(mediator);
        // 讓調停者知道所有同事
        mediator.setCdDriver(cdDriver);
        mediator.setCpu(cpu);
        mediator.setSoundCard(soundCard);
        mediator.setVideoCard(videoCard);
        // 開始看電影,把光盤放入光驅,光驅開始讀盤
        cdDriver.readCD();
    }
}

運行結果如下:

您正在觀看的是:One Piece
畫外音: 海賊王我當定了

調停者模式的優點

  • 松散耦合
    調停者模式通過把多個同事對象之間的交互封裝到調停者對象里面,從而使得同事對象之間松散耦合,基本上可以做到互補依賴。這樣一來,同事對象就可以獨立的變化和復用,而不再像以前那樣“牽一發而動全身”了。
  • 集中控制交互
    多個同事對象的交互,被封裝在調停者對象里面集中管理,使得這些交互行為發生變化的時候,只需要修改調停者對象就可以了,當然如果是已經做好的系統,那么就擴展調停者對象,而每個同事類不需要做修改。
  • 多對多變成一對多
    沒有使用調停者模式的時候,同事對象之前的關系通常是多對多的,引入調停者對象以后,調停者對象和同事對象的關系通常變成雙向的一對多,這會讓對象的關系更容易理解和實現。

調停者模式的缺點

調停者模式的一個潛在缺點是:過度集中化。如果同事對象的交互非常多,而且比較復雜,當這些復雜性全部集中到調停者的時候,會導致調停者對象變得十分復雜,而且難于管理和維護。

參考

《JAVA與模式》之調停者模式

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,967評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,273評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,870評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,742評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,527評論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,010評論 1 322
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,108評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,250評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,769評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,656評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,853評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,371評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,103評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,472評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,717評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,487評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,815評論 2 372

推薦閱讀更多精彩內容