圖解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
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...