設計模式 之 觀察者(Observer)模式

来源:http://www.cnblogs.com/mercuryli/archive/2016/03/15/5278897.html
-Advertisement-
Play Games

觀察者(observer)模式定義了一對多的依賴關係,讓多個觀察者對象能夠同時監聽某一主題對象。這個主題對象中的狀態發生改變時,就會通知所有的觀察者對象。 觀察者模式的結構圖: 結構中各個部分的含義: 源代碼: 抽象主題類(Subject):   具體主題類(ConcreteSubject):  


觀察者(observer)模式定義了一對多的依賴關係,讓多個觀察者對象能夠同時監聽某一主題對象。這個主題對象中的狀態發生改變時,就會通知所有的觀察者對象。

觀察者模式的結構圖:

結構中各個部分的含義:

  • 抽象主題類(Subject):它把所有對觀察者對象的引用都保存在一個聚集內,每個主題可以有任意多的觀察者。
  • 具體主題類(ConcreteSubject):具體主題,將有關狀態存入具體觀察者對象;當具體主題狀態改變時,向所有觀察者發出通知。
  • 抽象觀察者類(Observer):抽象觀察者,為所有的具體觀察者定義一個介面。
  • 具體觀察者類(ConcreteObserver):具體觀察者,實現抽象觀察者角色所要求的介面,以便更新本身的狀態

源代碼:

抽象主題類(Subject):

public interface Subject {
    public void registerObserver(Observer o); //增加觀察者
    public void removeObserver(Observer o);        //刪除觀察者
    public void notifyObserver(String newState);//通知觀察者
}

 

具體主題類(ConcreteSubject):

public class ConcreteSubject implements Subject{
    
    private ArrayList<Observer> observers; //觀察者集合
    
    public ConcreteSubject() {
        observers = new ArrayList<Observer>();
    }
    
    @Override
    public void registerObserver(Observer o) {
        observers.add(o);
    }

    @Override
    public void removeObserver(Observer o) {
        observers.remove(o);
    }

    @Override
    public void notifyObserver(String newstate) {
        for (Observer observer : observers) {
            observer.update(newstate);
        }
    }
    
    public void getChange(String newState){
        notifyObserver(newState);
    }
}

 

抽象觀察者類(Observer):

public interface Observer {
    public void update(String state);
}

 

抽象觀察者類(ConcreteObserver):

public class ConcreteObserver implements Observer{
        
    @Override
    public void update(String state) {
        System.out.println("更新後狀態為:"+ state);
    }

}

 

客戶端:

public class Client {
    public static void  main(String[] args) {
        ConcreteSubject s  = new ConcreteSubject();
        Observer observer = new ConcreteObserver();
        s.registerObserver(observer);
        s.getChange("新狀態");
    }
}

 

運行結果:

更新後狀態為:新狀態

 

 

具體情境舉例:氣象站的溫度變化更新:

抽象主題類(Subject):

public interface Subject {
    public void registerObserver(Observer o);
    public void removeObserver(Observer o);
    public void notifyObservers();
}

 

具體主題類:

public class CurrentConditionDisplay implements Observer,DisplayElement{

    private float temperature;
    private float humidity;
    private Subject weatherData;
    
    public CurrentConditionDisplay(Subject weatherData) {
        this.weatherData = weatherData;
        weatherData.registerObserver(this);
    }
    
    @Override
    public void display() {
        System.out.println("當前氣溫:"+ temperature + "F 濕度為:"+ humidity + "%");
    }

    @Override
    public void update(float temperature, float humidity, float pressure) {
        this.temperature = temperature;
        this.humidity = humidity;
        display();
    }

}

 

displayElement介面:

public interface DisplayElement {
    public void display();
}

 

抽象觀察者類(Observer):

public interface Observer {
    public void update(float temperature, float humidity, float pressure);
}

 

具體觀察者類(ConcreteObserver):

public class WeatherData implements Subject{

    private ArrayList<Observer> observers;    //觀察者
    //要更新的觀察者的信息
    private float temperature;
    private float humidity;
    private float pressure;
    
    public WeatherData() {
        observers = new ArrayList<Observer>();
    }
    
    @Override
    public void registerObserver(Observer o) {
        observers.add(o);
    }

    @Override
    public void removeObserver(Observer o) {
        int i = observers.indexOf(o);
        if (i >= 0 ) {
            observers.remove(i);
        }
    }

    @Override
    public void notifyObservers() {
//        for(int i = 0; i < observers.size(); i++){
//            Observer observer = (Observer) observers.get(i);
//            observer.update(temperature, humidity, pressure);
//        }
        for (Observer o : observers) {
            o.update(temperature, humidity, pressure);
        }
    }
    
    public void measurementsChanged(){
        notifyObservers();
    }
    
    public void setMeasurement(float temperature, float humidity, float pressure){
        this.temperature = temperature;
        this.humidity = humidity;
        this.pressure = pressure;
        measurementsChanged();
    }
}

 

客戶端:

public class WeatherStation {
    public static void main(String[] args){
        WeatherData weatherData = new WeatherData();
        
        CurrentConditionDisplay currentConditionDisplay = new CurrentConditionDisplay(weatherData);
        
        weatherData.setMeasurement(80, 65, 30.4f);
        weatherData.setMeasurement(82, 70, 29.2f);
        weatherData.setMeasurement(78, 90, 29.2f);
    }
}

 

運行結果:

當前氣溫:80.0F 濕度為:65.0%
當前氣溫:82.0F 濕度為:70.0%
當前氣溫:78.0F 濕度為:90.0%

 


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

-Advertisement-
Play Games
更多相關文章
  • 錯誤 C4996 初學C語言時,第一個接觸到的I/O函數便是scanf()了。但在高版本的 Visual Studio (包括但不限於2015、2013、2012)編譯代碼時,卻會出現意想不到的錯誤。有如下一段簡單的代碼: #include "stdio.h" int main(void) { in...
  • 繼承 概念: ① 繼承背後的思想就是基於已存在的類來構建新類; ② 當從已存在類繼承時,就重用了它的方法和屬性,還可以添加新的方法和屬性來定製新類以應對需求; ③ 當從其它類導出的類叫作子類,被導出的類叫作父類; ④ 在Java中,除了Object類外,所有類都是子類,都有唯一父類; ⑤ 繼承在OO
  • 對象A的創建過程: 1. 構造器實際上是靜態方法,所以,當首次創建對象A 或者 A類的靜態方法/靜態域首次被訪問時,Java解釋器查找類路徑,以位 A.class文件。 2. 載入A.class,有關靜態初始化的所有動作都會執行()。因此,靜態初始化只在Class對象首次載入的時候進行一次。 3. ...
  • 本篇分為兩部分:   Playground 就是提供一個可以即時編譯的類似 REPL 的環境,他為我們提供了一個順序執行的環境,在每次更改其中代碼後整個文件都會被重新編譯,並清空原來的狀態並運行。 NSTimer 在預設的 Playground 中是不會執行的: 在執行玩 NSTimer 語句之後,
  • 想了很久不知道怎麼入手 沒思路 不會寫    後來直接說不會寫   又給我出了一個 2+(3*2)+2 實現計算器輸入 輸出結果 讓我手寫出程式  蒙了 一塊面試的幾個人都沒做出面試題來 很是受打擊啊 醉了
  • 這是一本框架性的書,它從分散式網站架構設計需要考慮的內容出發,介紹了相關的技術。雖然每一部分講解不是特別深入,但是原理清晰,案例豐富,非常不錯。 現在的網站都越做越大,分散式是必須的選擇,通過這本書我看到了淘寶是怎麼做的,對於今後的工作很有幫助。 作者也是一位非常年輕的工程師,很不錯
  • 併發思想提煉(1)(理解併發,避免死鎖) 一直做伺服器後端和基礎組件平臺開發,常常用到併發,故簡單放些乾貨,一來算是總結,二來希望後人少走彎路, 寫到哪兒算哪兒,不定期更新。 1.    Introduction 先來明白一些概念。Concurrency併發和Multi-thread多線程不同 你在
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...