headfirst設計模式(2)—觀察者模式

来源:http://www.cnblogs.com/skyseavae/archive/2017/02/13/6379293.html
-Advertisement-
Play Games

定義 觀察者模式(有時又被稱為發佈(publish)-訂閱(Subscribe)模式,在此種模式中,一個目標物件管理所有相依於它的觀察者物件,並且在它本身的狀態改變時主動發出通知。這通常透過呼叫各觀察者所提供的方法來實現。此種模式通常被用來實現事件處理系統(摘自百度百科)。 關鍵詞:發佈-訂閱 為什 ...


 

定義

觀察者模式(有時又被稱為發佈(publish)-訂閱(Subscribe)模式,在此種模式中,一個目標物件管理所有相依於它的觀察者物件,並且在它本身的狀態改變時主動發出通知。這通常透過呼叫各觀察者所提供的方法來實現。此種模式通常被用來實現事件處理系統(摘自百度百科)。

關鍵詞:發佈-訂閱

為什麼只有一個關鍵詞?因為我覺得一個關鍵詞足夠說明問題了。觀察者模式適用於,一個對象改變時,需要通知一個或多個其他對象,而需要通知的對象的特點是:數量不清楚,類型不清楚(僅實現了一個通用介面),具體處理方式不清楚。

舉個例子來說明:

小花:你好,我開發過很多erp系統,是一位經驗豐富的女司機,現在想找一份java程式員的工作...(成熟穩重型)

獵頭:好的,我已經把你加入到我的程式員清單裡面了,不要打電話給我,我會通知你的(好萊塢原則)

小明:本人學識淵博、經驗豐富,代碼風騷、效率恐怖,c/c++、java、php無不精通,熟練掌握各種框架,深山苦練20餘年,一天只睡4小時,電話通知出bug後秒登vpn,千里之外定位問題,瞬息之間修複上線。 身體強壯、健步如飛,可連續編程100小時不休息,討論技術方案5小時不喝水,上至帶項目、出方案,下至盜賬號、威脅pm,什麼都能幹......(花式裝逼型)

獵頭:666,我已經把你加入到我的程式員清單裡面了,不要打電話給我,我會通知你的(小明和小花註冊為觀察者) 小花和小明繼續過著自己的日子,因為他們已經在獵頭的清單裡面了,有工作會收到通知的。 獵頭:小花同學,小明同學,這裡需要一個資深的全棧工程師,創業型,彈性工作制,股票期權,年終分紅...(通知所有觀察者) 小明:好的(隨後憑藉著小明的機智,獲得了這份工作) 獵頭:請把介紹費匯到我的銀行卡 又過了一段日子,小花憑藉著自己豐富的經驗,找到了工作,並沒有依靠獵頭,所以也就沒有什麼介紹費 小花:我已經找到工作了,請不要再給我發招聘信息了(移除觀察者) 10年之後... 小花娶妻生子,迎娶白富美,出任CEO,走上人生巔峰,孩子已經快要1米高了....(等等,好像有什麼地方不對) 小明由於天天加班,1天只睡4個小時,墳頭草已經1米高了... 當然這個是後話 以上就是訂閱和發佈的解釋,當然,我可以實話告訴你,這個和我下麵要貼的代碼並沒有什麼關係。

一個氣象監測應用的需求

概述:建立一個氣象觀測的應用,從氣象站獲取數據,並實時更新三個佈告板:目前狀況,天氣統計,天氣預報 目前狀況:溫度,濕度,氣壓 天氣統計:平均溫度,最低溫度,最高溫度 天氣預報:明天下雨嗎? 以下是具體實現,涉及到的設計原則: 1,針對抽象編程,不針對實現編程 2,多用組合少用繼承 3,開閉原則,對擴展開放,對修改關閉(所有設計模式都是圍繞這個終極目標來對不同場景進行設計的)
package observer;
/**
 * 觀察主題
 */
public interface Observable{
    public void addObserver(Observer observer);//添加觀察者
    public void removeObserver(Observer observer);//移除觀察者
    public void notifyObservers(WeatherData data);//通知所有觀察者
}

觀察主題也可以是抽象類,具體可以參考java.util.Observable,這裡就不展開來bb了。

那麼什麼時候選擇抽象類,什麼時候選擇介面呢?

我的理解是,這個就要看,代價高不高了,抽象類的好處就是,可以少寫代碼,實現復用。介面可以應對各種不同的變化,因為觀察者並不一定有共同使用的實現類,可能它們完全就是不同的東西。

這個地方我只有一個主題,我想用什麼就用什麼,就是這麼任性,如果以後出了第二個或者第三個主題,那麼就可以考慮具體是抽象類還是介面了,當然也不排除兩種都用的情況。當然這是後話,程式本來就是不斷變化的,適當的時候用適當的設計才是王道。

package observer;
/**
 * 觀察者
 */
public interface Observer {
    public abstract void update(WeatherData data);
}
package observer;
/**
 * 天氣數據
 */
public class WeatherData {
    private float temperature;
    private float humidity;
    private float pressure;
    public WeatherData(){
        
    }
    public WeatherData(float temperature, float humidity, float pressure){
        this.temperature = temperature;
        this.humidity = humidity;
        this.pressure = pressure;
    }
}

getter,setter省略,不然篇幅太長了

在Observer這個介面裡面,我的update()方法裡面帶了一個 WeatherData對象,為什麼不是Object呢,為什麼不是直接三個參數,溫度,濕度,氣壓呢?我當時想了很久(我也是一邊看書,一遍寫的,不是把所有的都學完了才來寫博客的,所以不懂的多了去了)

用Object的好處當然是不僅更新天氣可以用這個,以後更新其他什麼的也可以用,而且,如果我update方法裡面的需求有了新的變化,比如,我要更新時間,那麼,這個參數總不能放在WeatherData裡面吧,好不符合邏輯啊,強迫症怎麼受得了???

思來想去我的理解是這樣的:

1,這個是一個氣象站應用,那麼應該就是存氣象數據吧,這個總沒問題嘛

2,寫一個Object會不會被後面接替我工作的同事砍死啊,Object裡面是什麼,通過方法名根本就看不出來,必須去具體的代碼裡面看,萬一我邏輯有點複雜,註釋再少點,那他不就懵逼了?

3,不直接用三個參數也是為了以後的變化,萬一變成5個參數咋辦?改都改死人,而且要是5個都是float類型,好容易在傳的時候傳錯,發現得早還好,萬一傳錯了,但是又通過測試了怎麼辦呢?

肯定有人覺得這不可能,那我舉個例子

介面的定義是:

void test(int a, int b, int c);

實現是 void test(int a, int c, int b){...}

調用介面的時候,傳入的是test.test(a, c, b);那麼請問,這個能通過測試嗎?可以的。後面的人會掉坑裡嗎?我覺得他只要去改代碼,那麼就很有可能,嘿嘿嘿

package observer;
/**
 * 顯示
 */
public interface DisplayElement {
    void display();
}

這個介面沒什麼解釋的,就是顯示用的,下麵我要大段貼代碼了

package observer;
/**
 * 當前天氣狀況*/
public class CurrentConditionsDisplay implements DisplayElement, Observer {
    Observable observable;
    public CurrentConditionsDisplay(Observable observable) {
        this.observable = observable;
    }
    WeatherData data = new WeatherData();
    @Override
    public void display() {
        System.out.println("當前的天氣狀況: " + 
                            "溫度(℉):" + data.getTemperature() + " " + 
                            "濕度:" + data.getHumidity() + " " +
                            "氣壓:" + data.getPressure());
    }
    @Override
    public void update(WeatherData data) {
        this.data = data;
        display();
    }
}
package observer;
/**
 * 天氣統計
 *
 */
public class StatisticsDisplay implements DisplayElement, Observer {

    private Float average;
    private Float highest;
    private Float lowest;
    @Override
    public void display() {
        System.out.println("天氣統計:" +
                            "平均溫度(℉):" + average + " " +
                            "最高溫度(℉):" + highest + " " +
                            "最低溫度(℉):" + lowest);
    }
    @Override
    public void update(WeatherData data) {
        float temperature = data.getTemperature();
        average = null == average ? temperature : 
                        (average + temperature) / 2;
        highest = null == highest ? temperature : 
                        (highest > temperature ? highest : temperature);
        lowest = null == lowest ? temperature : 
                        (lowest > temperature ? temperature : lowest);
        display();
    }

}
package observer;
/**
 * 天氣預報*/
public class ForecastDisplay implements DisplayElement, Observer {
    private float pressure;
    @Override
    public void update(WeatherData data) {
        pressure = data.getPressure();
        display();
    }
    @Override
    public void display() {
        System.out.println("天氣預測:" + forecast());
    }
    private static final float INFRABAR = 28.5f;//氣壓低就會下雨
    private String forecast(){
        return INFRABAR < pressure ? "明天不下雨" : "明天要下雨";
    }
}
package observer;

import java.util.ArrayList;
import java.util.List; /** * 天氣主題 * */ public class Weather implements Observable { private List<Observer> observers = new ArrayList<>();; public void addObserver(Observer observer) { observers.add(observer); } public void removeObserver(Observer observer) { observers.remove(observer); } public void notifyObservers(WeatherData data) { for (Observer observer : observers) observer.update(data); } }
package observer;
/**
 * 測試
 * @author Skysea
 *
 */
public class Client {
    
    public static void main(String[] args) {
        Observable observable = new Weather();
        Observer currentCondition = new CurrentConditionsDisplay(observable);
        Observer statistics = new StatisticsDisplay();
        Observer forecast = new ForecastDisplay();
        observable.addObserver(currentCondition);
        observable.addObserver(statistics);
        observable.addObserver(forecast);
        observable.notifyObservers(new WeatherData(80, 65, 30.4f));
        observable.notifyObservers(new WeatherData(82, 70, 29.2f));
        observable.notifyObservers(new WeatherData(78, 90, 25.1f));
    }
}

運行結果:

在測試代碼中,添加觀察者到主題這些代碼,也是可以放在觀察者中去實現的,在初始化的時候就把自己加入到主題的列表中

在實際的使用中,如使用Spring的依賴註入之類的,還是很好用的,也可以省很多事情,最後的交互方式大概是這個樣子

package observer;
/**
 * 測試2
 */
public class Client2 {
@Autowired private Observable observable;
public static void main(String[] args) { observable.notifyObservers(new WeatherData(80, 65, 30.4f)); observable.notifyObservers(new WeatherData(82, 70, 29.2f)); observable.notifyObservers(new WeatherData(78, 90, 25.1f)); } }

嗯,沒錯,什麼初始化都沒有了,觀察者在啟動的時候就註入到主題中了,只需要調用主題的介面就好了,而主題也僅僅是持有了一個List<Observer> observers;具體的觀察者是誰它也不知道,但是它們還是可以相互交互,是不是很酷?

好了,我bb完了,本系列的大部分所有例子,都是在看headfirst設計模式時,看到的,並加入自己的理解寫下來的(當然我也編了一些),寫這段話的目的也是為了避免一些不必要的誤會,以後的每期我都會複製這段話,哈哈哈


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

-Advertisement-
Play Games
更多相關文章
  • struts2框架 如果你之前在MVC模式的時候一直都是通過servlet,獲取和返回數據,那麼現在開始學習struts2框架, Struts是一個實現MVC設計模式的優秀的框架。它的許多優點我就不說了。 我用自己做的一張圖說明servlet和struts2的區別。 寫一個最基本的開發步驟,完成開發 ...
  • 一.騰訊優圖 1.開發者地址:http://open.youtu.qq.com/welcome/developer 2.接入流程:按照開發者頁面的接入流程接入之後,創建應用即可獲得所需的AppID、SecretID和SecretKey這是進行介面調用必須的憑證 3.測試流程: 3.1.測試可以直接調 ...
  • 這裡並未涉及到JSR181Annotations的相關應用,具體的三種方式如下 ①通過WSDL地址來創建動態客戶端②通過服務端提供的介面來創建客戶端③使用Ant通過WSDL文件來生成客戶端 第一種方式:通過WSDL地址來創建動態客戶端 view plainprint? ...
  • 列印thinkphp中的sql語句 var_dump($repair->fetchSql(true)->where(array('cuername' =>$cuername))->order('applytime desc')->limit($page1*$listRows,$listRows)-> ...
  • 歡迎大家來咨詢海南七星彩打獎系統,系統可以支持手機下註的南方海南,湛江七星彩投註網站系統,也可以出租,出售等渠道,或者定製,可以選擇支持手機下註或者不支持,需要定製的,可以私信,扣扣:1930-1335-70 本截圖只作為演示,如有需要定製,購買,請聯繫客服購買正版授權使用。 會員演示圖: 代理演示 ...
  • 1 import java.util.*; 2 class CalendarTest 3 { 4 /*先輸出提示語句,並接受用戶輸入的年、月。 5 根據用戶輸入的年,先判斷是否是閏年。 6 根據用戶輸入的年份來判斷月的天數。 7 用迴圈計算用戶輸入的年份距1900年1月1日的總天數。 8 用迴圈計算... ...
  • mybatis和hibernate之間的對比。及應用場景的介紹 ...
  • sso單點登錄系統的最全的搭建方法,只要你按我的步驟來,就可以成功的搭建出你的sso單點登錄系統。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...