圖解Java設計模式之中介者模式

来源:https://www.cnblogs.com/haizai/archive/2020/04/02/12622618.html
-Advertisement-
Play Games

圖解Java設計模式之中介者模式 智能家庭項目 傳統方案解決智能家庭管理問題 中介者模式基本介紹 中介者模式的原理類圖 中介者模式 - 智能家庭的操作流程 中介者模式的註意事項和細節 智能家庭項目 1)智能家庭包括各種設備,鬧鐘、咖啡機、電視機、窗帘等2)主人要看電視時,各個設備可以協同工作,自動完 ...


圖解Java設計模式之中介者模式

 

智能家庭項目

1)智能家庭包括各種設備,鬧鐘、咖啡機、電視機、窗帘等
2)主人要看電視時,各個設備可以協同工作,自動完成看電視的準備工作,比如流程為 :鬧鈴響起 - 》咖啡機開始做咖啡 -》窗帘自動落下 -》電視機開始播放

傳統方案解決智能家庭管理問題

在這裡插入圖片描述
1)當各電器對象有多種狀態改變時,相互之間的調用關係會比較複雜。
2)各個電器對象彼此聯繫,你中有我,我中有你,不利於松耦合。
3)各個電器對象之間所傳遞的消息(參數),容易混亂
4)當系統增加一個新的電器對象時,或者執行流程改變時,代碼的可維護性、擴展性都不理想。

中介者模式基本介紹

1)中介者模式(Mediator Pattern),用一個中介對象來封裝一系列的對象交互。
中介者使各個對象不需要顯式地相互引用,從而使其耦合鬆散,而且可以獨立地改變它們之間的交互。
2)中介者模式屬於行為型模式,使代碼易於維護。
3)比如MVC模式,C(Controller控制器)是M(Model模型)和V(View視圖)的中介者,在前後端交互時起到來中間人的作用。

中介者模式的原理類圖

在這裡插入圖片描述
對原理類圖的說明 :
1)Mediator 就是抽象中介者,定義類同事對象到中介者對象的介面。
2)Colleague 是抽象同事類。
3)ConcreteMediator 具體的中介者對象,實現抽象方法,他需要知道所有的具體的同事類,即以一個集合來管理HashMap,並接收某個同事對象消息,完成相應的任務。
4)ConcreteColleague 具體的同事類,會有很多,每個同事只知道自己的行為,而不瞭解其他同事類的行為(方法),但是他們都是依賴中介者對象。

中介者模式 - 智能家庭的操作流程

在這裡插入圖片描述

  1. 創建ConcreMediator 對象
  2. 創建各個同事類對象,比如 :Alarm、CoffeeMachine、TV
  3. 在創建同事類對象的時候,就直接通過構造器,加入到colleagueMap
  4. 同事類對象,可以調用sendMessage,最終會去調用ConcreteMediator的getMessage方法
  5. getMessage 會根據接收到的同事對象發出的消息來協調調用其他的同事對象,完成任務
  6. 可以看到getMessage 是核心方法,完成相應任務
package com.example.demo.mediator;

public abstract class Mediator {

	//將給中介者對象,加入到集合中
	public abstract void Register(String colleagueName, Colleague colleague);
	//接收消息, 具體的同事對象發出
	public abstract void GetMessage(int stateChange, String colleagueName);
	public abstract void SendMessage();
}
package com.example.demo.mediator;

public abstract class Colleague {

	private Mediator mediator; 
	public String name;
	public Colleague(Mediator mediator, String name) {
		this.mediator = mediator; 
		this.name = name;
	}
	public Mediator GetMediator() { 
		return this.mediator;
	}
	public abstract void SendMessage(int stateChange);
}
package com.example.demo.mediator;

public class Alarm extends Colleague {

	public Alarm(Mediator mediator, String name) {
		super(mediator, name);
		// TODO Auto-generated constructor stub
		//在創建 Alarm 同事對象時,將自己放入到 ConcreteMediator 對象中[集合] 
		mediator.Register(name, this);
	}

	public void SendAlarm(int stateChange) { 
		SendMessage(stateChange);
	}
	
	@Override
	public void SendMessage(int stateChange) {
		// TODO Auto-generated method stub
		//調用的中介者對象的 getMessage 
		this.GetMediator().GetMessage(stateChange, this.name);
	}

}
package com.example.demo.mediator;

public class CoffeeMachine extends Colleague {

	public CoffeeMachine(Mediator mediator, String name) {
		super(mediator, name);
		// TODO Auto-generated constructor stub
		mediator.Register(name, this);
	}

	@Override
	public void SendMessage(int stateChange) {
		// TODO Auto-generated method stub
		this.GetMediator().GetMessage(stateChange, this.name);
	}
	
	public void StartCoffee() { 
		System.out.println("It's time to startcoffee!");
	}
	public void FinishCoffee() {
		System.out.println("After 5 minutes!"); 
		System.out.println("Coffee is ok!"); 
		SendMessage(0);
	}

}
package com.example.demo.mediator;

public class Curtains extends Colleague {

	public Curtains(Mediator mediator, String name) {
		super(mediator, name);
		// TODO Auto-generated constructor stub
		mediator.Register(name, this);
	}

	@Override
	public void SendMessage(int stateChange) {
		// TODO Auto-generated method stub
		this.GetMediator().GetMessage(stateChange, this.name);
	}
	
	public void UpCurtains() {
		System.out.println("I am holding Up Curtains!");
	}

}
package com.example.demo.mediator;

public class TV extends Colleague {

	public TV(Mediator mediator, String name) {
		super(mediator, name);
		// TODO Auto-generated constructor stub
		mediator.Register(name, this);
	}

	@Override
	public void SendMessage(int stateChange) {
		// TODO Auto-generated method stub
		this.GetMediator().GetMessage(stateChange, this.name);
	}
	
	public void StartTv() {
		// TODO Auto-generated method stub 
		System.out.println("It's time to StartTv!");
	}
	public void StopTv() {
		// TODO Auto-generated method stub 
		System.out.println("StopTv!");
	}

}
package com.example.demo.mediator;

import java.util.HashMap;

public class ConcreteMediator extends Mediator {
	
	//集合,放入所有的同事對象
	private HashMap<String, Colleague> colleagueMap; 
	private HashMap<String, String> interMap;

	public ConcreteMediator() {
		colleagueMap = new HashMap<String, Colleague>(); 
		interMap = new HashMap<String, String>();
	}

	@Override
	public void Register(String colleagueName, Colleague colleague) {
		colleagueMap.put(colleagueName, colleague);
		// TODO Auto-generated method stub
		if (colleague instanceof Alarm) { 
			interMap.put("Alarm", colleagueName);
		} else if (colleague instanceof CoffeeMachine) { 
			interMap.put("CoffeeMachine", colleagueName);
		} else if (colleague instanceof TV) { 
			interMap.put("TV", colleagueName);
		} else if (colleague instanceof Curtains) {
			interMap.put("Curtains", colleagueName); 
		}
	}

	//具體中介者的核心方法
	//1. 根據得到消息,完成對應任務
	//2. 中介者在這個方法,協調各個具體的同事對象,完成任務 
	@Override
	public void GetMessage(int stateChange, String colleagueName) {
		// TODO Auto-generated method stub
		//處理鬧鐘發出的消息
		if (colleagueMap.get(colleagueName) instanceof Alarm) {
			if (stateChange == 0) {
				((CoffeeMachine) (colleagueMap.get(interMap
						.get("CoffeeMachine")))).StartCoffee();
				((TV) (colleagueMap.get(interMap.get("TV")))).StartTv();
			} else if (stateChange == 1) {
				((TV) (colleagueMap.get(interMap.get("TV")))).StopTv();
			}
		 else if (colleagueMap.get(colleagueName) instanceof CoffeeMachine) { 
			 ((Curtains) (colleagueMap.get(interMap.get("Curtains")))).UpCurtains();
		 }
		} else if (colleagueMap.get(colleagueName) instanceof TV) {
			//如果 TV 發現消息
		} else if (colleagueMap.get(colleagueName) instanceof Curtains) { 
			//如果是以窗帘發出的消息,這裡處理...
		}
	 
	}
	@Override
	public void SendMessage() {
		// TODO Auto-generated method stub 
	}

}
package com.example.demo.mediator;

import java.util.HashMap;

public class Client {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		//創建一個中介者對象
		Mediator mediator = new ConcreteMediator();
		//創建 Alarm 並且加入到 ConcreteMediator 對象的 HashMap 
		Alarm alarm = new Alarm(mediator, "alarm");
		//創建了 CoffeeMachine 對象,並 且加入到 ConcreteMediator 對象的 HashMap 
		CoffeeMachine coffeeMachine = new CoffeeMachine(mediator,"coffeeMachine");
		//創建 Curtains , 並 且加入到 ConcreteMediator 對象的 HashMap 
		Curtains curtains = new Curtains(mediator, "curtains");
		TV tV = new TV(mediator, "TV");
		//讓鬧鐘發出消息 
		alarm.SendAlarm(0); 
		coffeeMachine.FinishCoffee();
		alarm.SendAlarm(1);
	}

}

中介者模式的註意事項和細節

1)多個類相互耦合,會形成網站結構,使用中介者模式將網狀結構分離為星型結構,進行解耦。
2)減少類間依賴,降低類耦合,符合迪米特法則。
3)中介者承擔類較多的責任,一旦中介者出現了問題,整個系統就會受到影響。
4)如果設計不當,中介者對象本身變得過於複雜,這點在實際使用時,要特別註意。


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

-Advertisement-
Play Games
更多相關文章
  • 可視化工具D3.js,就不贅述他的簡介與下載了,官網上看,直接上教程: 可視化工具D3.js教程 入門 (第一章)—— hello world 可視化工具D3.js教程 入門 (第二章)—— 選擇元素與數據綁定 可視化工具D3.js教程 入門 (第三章)—— 理解 Update Enter Exit ...
  • 還是那句話,有時候你可能沒有Jquery可用,那麼我們可以借用原生JS中Dom對象的新方法:querySelector和querySelectorAll 實現Jquery篩選器同樣的篩選節點的效果,下麵列舉一些示例: 1-直接從頁面中篩選元素: id選擇器: var myidDom=document ...
  • 有時候,我們無法藉助熟悉的jquery發起請求,原生JS里是支持fetch函數的,這是個高度封裝的方法,幫助我們做了很多底層的封裝,下麵列舉一些發起請求的示例: 1-發起Get請求: //httpGet請求 var httpGet = async function (getUrl) { var op ...
  • 偶然間遇到,需要在JS中解決類似於C#中的線程休眠問題,JS有Promise對象,可以幫助我們實現這一點,網上有很多類似文章,我這裡列舉一個使用示例: 定義休眠方法--使用Promise創建一個非同步可等待的方法: //設置JS休眠一定時間後執行的動作 var sleep = function (mi ...
  • 一、GOF 23種設計模式簡介 設計模式其實是一門藝術。設計模式來源於生活,不要為了套用設計模式而去使用設計模式。設計模式是在我們迷茫時提供的一種解決問題的方案,或者說用好設計模式可以防範於未然。自古以來,在我們人生迷茫時,我們往往都會尋求幫助,或上門咨詢,或查經問典。就在幾千年前,孔夫子就教給了我 ...
  • 前言 本文章針對Java課程前三次PTA作業進行總結 一.作業過程總結 1.總結三次作業之間的知識迭代關係:關於前三次作業之間的迭代關係,第一次作業主要是關於Java的最基礎的練習,演算法都可以直接再main函數裡面實現,初步瞭解Java的一些基本的操作,輸入輸出,至於其它語句和C語言都是一樣的。 第 ...
  • 圖解Java設計模式之解釋器模式 四則運算問題 傳統方案解決四則運算問題分析 解釋器模式基本介紹 解釋器模式來實現四則 解析器模式在Spring框架中的源碼分析 解釋器模式的註意事項和細節 四則運算問題 通過解釋器模式來實現四則運算,如計算 a + b + c 的值,具體要求1)先輸入表達式的形式, ...
  • 圖解Java設計模式之備忘錄模式 游戲角色狀態恢復問題 傳統方案解決游戲角色恢復 傳統的方式的問題分析 備忘錄模式基本介紹 備忘錄模式原理類圖 游戲角色恢復狀態實例 備忘錄模式的註意事項和細節 游戲角色狀態恢復問題 游戲角色有攻擊力和防禦力,在大戰Boss前保存自身的狀態(攻擊力和防禦力),當大戰B ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...