Observer設(shè)計(jì)模式范例詳解
Observer設(shè)計(jì)模式說(shuō)明
假設(shè)我們有個(gè)高檔的熱水器,我們給它通上電,當(dāng)水溫超過(guò)95度的時(shí)候:1、揚(yáng)聲器會(huì)開(kāi)始發(fā)出語(yǔ)音,告訴你水的溫度;2、液晶屏也會(huì)改變水溫的顯示,來(lái)提示水已經(jīng)快燒開(kāi)了。
現(xiàn)在我們需要寫個(gè)程序來(lái)模擬這個(gè)燒水的過(guò)程,我們將定義一個(gè)類來(lái)代表熱水器,我們管它叫:Heater,它有代表水溫的字段,叫做temperature;當(dāng)然,還有必不可少的給水加熱方法BoilWater(),一個(gè)發(fā)出語(yǔ)音警報(bào)的方法MakeAlert(),一個(gè)顯示水溫的方法,ShowMsg()。
- namespace Delegate {
- class Heater {
- private int temperature; // 水溫
- // 燒水
- public void BoilWater() {
- for (int i = 0; i < = 100; i++) {
- temperature = i;
- if (temperature > 95) {
- MakeAlert(temperature);
- ShowMsg(temperature);
- }
- }
- }
- // 發(fā)出語(yǔ)音警報(bào)
- private void MakeAlert(int param) {
- Console.WriteLine("Alarm:嘀嘀嘀,水已經(jīng) {0} 度了:" , param);
- }
- // 顯示水溫
- private void ShowMsg(int param) {
- Console.WriteLine("Display:水快開(kāi)了,當(dāng)前溫度:{0}度。" , param);
- }
- }
- class Program {
- static void Main() {
- Heater ht = new Heater();
- ht.BoilWater();
- }
- }
- }
Observer設(shè)計(jì)模式簡(jiǎn)介
上面的例子顯然能完成我們之前描述的工作,但是卻并不夠好?,F(xiàn)在假設(shè)熱水器由三部分組成:熱水器、警報(bào)器、顯示器,它們來(lái)自于不同廠商并進(jìn)行了組裝。那么,應(yīng)該是熱水器僅僅負(fù)責(zé)燒水,它不能發(fā)出警報(bào)也不能顯示水溫;在水燒開(kāi)時(shí)由警報(bào)器發(fā)出警報(bào)、顯示器顯示提示和水溫。
這時(shí)候,上面的例子就應(yīng)該變成這個(gè)樣子:
- // 熱水器
- public class Heater {
- private int temperature;
- // 燒水
- private void BoilWater() {
- for (int i = 0; i < = 100; i++) {
- temperature = i;
- }
- }
- }
- // 警報(bào)器
- public class Alarm{
- private void MakeAlert(int param) {
- Console.WriteLine("Alarm:嘀嘀嘀,水已經(jīng) {0} 度了:" , param);
- }
- }
- // 顯示器
- public class Display{
- private void ShowMsg(int param) {
- Console.WriteLine("Display:水已燒開(kāi),當(dāng)前溫度:{0}度。" , param);
- }
- }
這里就出現(xiàn)了一個(gè)問(wèn)題:如何在水燒開(kāi)的時(shí)候通知報(bào)警器和顯示器?在繼續(xù)進(jìn)行之前,我們先了解一下Observer設(shè)計(jì)模式,Observer設(shè)計(jì)模式中主要包括如下兩類對(duì)象:
Subject:監(jiān)視對(duì)象,它往往包含著其他對(duì)象所感興趣的內(nèi)容。在本范例中,熱水器就是一個(gè)監(jiān)視對(duì)象,它包含的其他對(duì)象所感興趣的內(nèi)容,就是temprature字段,當(dāng)這個(gè)字段的值快到100時(shí),會(huì)不斷把數(shù)據(jù)發(fā)給監(jiān)視它的對(duì)象。
Observer:監(jiān)視者,它監(jiān)視Subject,當(dāng)Subject中的某件事發(fā)生的時(shí)候,會(huì)告知Observer,而Observer則會(huì)采取相應(yīng)的行動(dòng)。在本范例中,Observer有警報(bào)器和顯示器,它們采取的行動(dòng)分別是發(fā)出警報(bào)和顯示水溫。
在本例中,事情發(fā)生的順序應(yīng)該是這樣的:
警報(bào)器和顯示器告訴熱水器,它對(duì)它的溫度比較感興趣(注冊(cè))。
熱水器知道后保留對(duì)警報(bào)器和顯示器的引用。
熱水器進(jìn)行燒水這一動(dòng)作,當(dāng)水溫超過(guò)95度時(shí),通過(guò)對(duì)警報(bào)器和顯示器的引用,自動(dòng)調(diào)用警報(bào)器的MakeAlert()方法、顯示器的ShowMsg()方法。
類似這樣的例子是很多的,GOF對(duì)它進(jìn)行了抽象,稱為Observer設(shè)計(jì)模式:Observer設(shè)計(jì)模式是為了定義對(duì)象間的一種一對(duì)多的依賴關(guān)系,以便于當(dāng)一個(gè)對(duì)象的狀態(tài)改變時(shí),其他依賴于它的對(duì)象會(huì)被自動(dòng)告知并更新。Observer模式是一種松耦合的設(shè)計(jì)模式。
實(shí)現(xiàn)范例的Observer設(shè)計(jì)模式
我們之前已經(jīng)對(duì)委托和事件介紹很多了,現(xiàn)在寫代碼應(yīng)該很容易了,現(xiàn)在在這里直接給出代碼,并在注釋中加以說(shuō)明。
- using System;
- using System.Collections.Generic;
- using System.Text;
- namespace Delegate {
- // 熱水器
- public class Heater {
- private int temperature;
- public delegate void BoilHandler(int param); //聲明委托
- public event BoilHandler BoilEvent; //聲明事件
- // 燒水
- public void BoilWater() {
- for (int i = 0; i < = 100; i++) {
- temperature = i;
- if (temperature > 95) {
- if (BoilEvent != null) { //如果有對(duì)象注冊(cè)
- BoilEvent(temperature); //調(diào)用所有注冊(cè)對(duì)象的方法
- }
- }
- }
- }
- }
- // 警報(bào)器
- public class Alarm {
- public void MakeAlert(int param) {
- Console.WriteLine("Alarm:嘀嘀嘀,水已經(jīng) {0} 度了:", param);
- }
- }
- // 顯示器
- public class Display {
- public static void ShowMsg(int param) { //靜態(tài)方法
- Console.WriteLine("Display:水快燒開(kāi)了,當(dāng)前溫度:{0}度。", param);
- }
- }
- class Program {
- static void Main() {
- Heater heater = new Heater();
- Alarm alarm = new Alarm();
- heater.BoilEvent += alarm.MakeAlert; //注冊(cè)方法
- heater.BoilEvent += (new Alarm()).MakeAlert; //給匿名對(duì)象注冊(cè)方法
- heater.BoilEvent += Display.ShowMsg; //注冊(cè)靜態(tài)方法
- heater.BoilWater(); //燒水,會(huì)自動(dòng)調(diào)用注冊(cè)過(guò)對(duì)象的方法
- }
- }
- }
輸出為:
- Alarm:嘀嘀嘀,水已經(jīng) 96 度了:
- Alarm:嘀嘀嘀,水已經(jīng) 96 度了:
- Display:水快燒開(kāi)了,當(dāng)前溫度:96度。
- // 省略...
【編輯推薦】