設計模式之觀察者模式(observer pattern)

来源:https://www.cnblogs.com/yssjun/archive/2019/06/29/11107038.html
-Advertisement-
Play Games

觀察者模式主要用於處理對象間的一對多的關係,是一種對象行為模式。該模式的實際應用場景比較容易確認,當一個對象狀態發生變化時,所有該對象的關註者均能收到狀態變化通知,以進行相應的處理。本文希望通過簡單的介紹和分析,能讓讀者對觀察者模式有一個簡單直觀的認識和感知,以便在實際開發中根據需要靈活運用。 1. ...


觀察者模式主要用於處理對象間的一對多的關係,是一種對象行為模式。該模式的實際應用場景比較容易確認,當一個對象狀態發生變化時,所有該對象的關註者均能收到狀態變化通知,以進行相應的處理。
本文希望通過簡單的介紹和分析,能讓讀者對觀察者模式有一個簡單直觀的認識和感知,以便在實際開發中根據需要靈活運用。

1. 目的

建立對象間一對多的關聯關係,並能使一個對象的變化被所有關聯對象感知。

2. 動機

建立一套低耦合的消息觸發機制。

3. 優缺點

優點:

  1. 被觀察者和觀察者之間是抽象耦合的;
  2. 耦合度較低,兩者之間的關聯僅僅在於消息的通知;
  3. 被觀察者無需關心他的觀察者;
  4. 支持廣播通信;

缺點:

  1. 觀察者只知道被觀察對象發生了變化,但不知變化的過程和緣由;
  2. 觀察者同時也可能是被觀察者,消息傳遞的鏈路可能會過長,完成所有通知花費時間較多;
  3. 如果觀察者和被觀察者之間產生迴圈依賴,或者消息傳遞鏈路形成閉環,會導致無限迴圈;

4. 應用場景

  • 需要在系統中建立一個單項廣播的觸發機制;
  • 系統中某個對象的行為會影響若幹其他對象;
  • 對象之間的關聯關係可以在運行時動態的建立與撤銷;
  • 對象之間的關聯關係呈現出一種樹狀結構;

5.  原理

下麵是GoF介紹的典型的類觀察者模式的UML類圖:

Subject:

 抽象被觀察者,僅提供註冊和刪除觀察者對象的介面聲明。

ConcreteSubject:

 具體被觀察者對象,該對象中收集了所有需要被通知的觀察者,並可以動態的增刪集合中的觀察者。當其狀態發生變化時會通知所有觀察者對象。

Observer:

 抽象觀察者,為所有觀察者定義獲得通知的統一介面;

ConcreteObserver:

 觀察者對象,其關註對象為Subject,能接受Subject變化時發出的通知並更新自身狀態。

6.實現

接下來先將上面的UML類圖轉換為具體的代碼,然後在舉一個具體的例子來看一下其應用。

抽象被觀察者類:Subject

public interface Subject {
    public void setState(int state);
    public int getState();
    public void attach(Observer obs);
    public void detach(Observer obs);
    public void notify(String msg);
}

 抽象觀察者類:Observer

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

具體被觀察者類:ConcreteSubject

public class ConcreteSubject implements Subject {
    
    private List<Observer> observerList = new ArrayList<Observer>();
    private int state;

    @Override
    public void setState(int state) {
        this.state = state;        
        notify("new state: " + state);
    }

    @Override
    public int getState() {
        // TODO Auto-generated method stub
        return 0;
    }
    
    @Override
    public void attach(Observer obs) {
        // TODO Auto-generated method stub
        observerList.add(obs);
    }

    @Override
    public void detach(Observer obs) {
        // TODO Auto-generated method stub
        observerList.remove(obs);
    }

    @Override
    public void notify(String msg) {
        // TODO Auto-generated method stub
        for (Observer obs: observerList) {
            obs.update(msg);
        }
    }
}

具體觀察者類:ConcreteObserver

public class ConcreteObserver implements Observer {

    @Override
    public void update(String msg) {
        // TODO Auto-generated method stub
        System.out.println("ConcreteObserver receive notify msg: " + msg);
    }

}

演示:

public class Demo {
    public static void main(String[] args) {
        ConcreteObserver obs = new ConcreteObserver();
        ConcreteSubject sub = new ConcreteSubject();
        sub.attach(obs);
        sub.setState(666);
        sub.notify("just test subject notify function!");
    }
}

結果:

ConcreteObserver receive notify msg: new state: 666
ConcreteObserver receive notify msg: just test subject notify function!

7.實例

我們以一個更加實際的例子——商品價格的變動來體會一下觀察者模式的用途。

在網上購物的時候,商品一般都有一個價格變動通知,前提是我們關註了該商品。

這裡我們稍微變通一下,只有當關註的商品價格下降,且低於用戶期望購買價格的時候,才會給用戶發送一條商品降價的簡訊通知。

下麵分別定義每個類:

產品抽象類:Product

public interface Product {
    public void setPrice(int price);
    public int getPrice();
    public void follow(User user);
    public void unfollow(User user);
    public void notifyLowPrice();
}

用戶抽象類:User

public interface User {
    public boolean isExpectedPrice(int price);
    public void shortMSG(String msg);
}

商品筆記本電腦:Laptop

public class Laptop implements Product {
    
    private List<User> followList = new ArrayList<User>();
    private int curPrice;

    @Override
    public void setPrice(int price) {
        curPrice = price;
        System.out.println("set laptop price: " + price);
        notifyLowPrice();
    }

    @Override
    public int getPrice() {
        return curPrice;
    }
    
    @Override
    public void follow(User user) {
        followList.add(user);
    }

    @Override
    public void unfollow(User user) {
        followList.remove(user);
    }

    @Override
    public void notifyLowPrice() {
        String msg = "" + curPrice;
        for (User user: followList) {
            if (user.isExpectedPrice(curPrice)) {
                user.shortMSG(msg);
            }
        }
    }
}

關註筆記本電腦用戶類:LaptopBuyer

public class LaptopBuyer implements User {
    private int expectedPrice;
    private String userName;
    public LaptopBuyer(String userName, int expectedPrice) {
        this.userName = userName;
        this.expectedPrice = expectedPrice;
    }

    @Override
    public boolean isExpectedPrice(int curPrice) {
        // TODO Auto-generated method stub
        return curPrice <= expectedPrice;
    }

    @Override
    public void shortMSG(String msg) {
        // TODO Auto-generated method stub
        System.out.println("Your follow product have a low price: " + msg + " TO:" + userName);
    }

}

演示:

public class Demo {
    public static void main(String[] args) {
        LaptopBuyer Alice = new LaptopBuyer("Alice", 6000);
        LaptopBuyer Jack = new LaptopBuyer("Jack", 6500);
        Laptop laptop = new Laptop();
        laptop.follow(Alice);
        laptop.follow(Jack);
        laptop.setPrice(7000);
        laptop.setPrice(6500);
        laptop.setPrice(6000);
        laptop.unfollow(Jack);
        laptop.setPrice(5999);
        laptop.setPrice(6099);
    }
}

結果:

set laptop price: 7000
set laptop price: 6500
Your follow product have a low price: 6500 TO:Jack
set laptop price: 6000
Your follow product have a low price: 6000 TO:Alice
Your follow product have a low price: 6000 TO:Jack
set laptop price: 5999
Your follow product have a low price: 5999 TO:Alice
set laptop price: 6099

上面的這個例子是一個能夠很好地解釋觀察者模式的一個實際用途。

8. 總結

相比較與觀察者模式,我們或許有許多獲取另外一個對象狀態的方式,比如,常見的輪詢方式,或者僅僅在需要的時候去查一下對方的狀態等,不過觀察者模式有其特殊的用途,而且更加靈活。

該模式原理比較簡單直接,但是實際使用過程中需要考慮一些細節問題:

  • 何時通知?
  • 有誰觸發通知?
  • 觀察者是關註狀態變化的次數還是最終的狀態?
  • 如果消息通知被阻塞,應該怎麼辦?
  • 是否可以改為非同步消息通知?

上面這些都是實際使用時應該考慮的。考慮清楚這些細節才能更靈活的應用該模式解決實際問題。

參考:

GoF《Design Patterns: Elements of Reusable Object-Oriented Software》

https://www.runoob.com/design-pattern/observer-pattern.html


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

-Advertisement-
Play Games
更多相關文章
  • [2019.06.29 學習筆記1] 1.定義文檔主體。 2.常見屬性background:設置文檔背景圖片,不設置其他的情況預設平鋪。 3.常見屬性bgcolor:設置文檔背景顏色。 ...
  • 今天在找資料的時候,空調維修發現了一篇關於“javascript與jQuery”文章。講得還不錯,拿來給大家一起分享下: 其實很多初學JS的朋友們,都會糾結一個問題?是什麼問題呢? “學習js到底是先學原生JS還是jQuery呢?” 其實在我認為:你首先要明白什麼是javascript簡稱(原生JS ...
  • 相信大家對vue.js這個前端框架有了一定的瞭解。想必也想把Vue急切的運用在項目中,看看它的魅力到底有多大?別急,今天我會滿足大家的想法。 我們一起來看看“Webpack+Vue”的開發模式相比以往老項目(Gulp+jQuery)的開發模式的魅力在哪裡。 一、配置開發環境 1、先安裝Node和We ...
  • 1、效果圖如下: 2、源碼如下: html部分: js部分: css部分: 兩個文件,carts.css 和 reset.css cart.css部分: 源碼地址: 鏈接:https://pan.baidu.com/s/1HV1zk3QsJti8yyJ1gNBWnQ 提取碼:skhx 如鏈接過期了私 ...
  • 這些東西有些比較常用,有些僅知道個名稱,但無論是熟悉還是陌生的,要比較精確地解釋這些東西,是有一定的難度,可這些東西對前端開發非常重要,還是需要有明確的概念。 PS:內容點到即止,不然一個東西一篇文章都寫不完。 整體認識 在學習方法上,對於互有聯繫的東西,有一個整體的印象,比起一個個瑣碎的認識更為重 ...
  • 一、項目介紹 基於react+react-dom+react-router-dom+redux+react-redux+webpack2.0+nodejs等技術混合開發的仿微信web端聊天室reactWebChat項目,實現了聊天記錄右鍵菜單、發送消息、表情(動圖),圖片、視頻預覽,瀏覽器截圖粘貼發 ...
  • 一、生命周期鉤子(函數): 1、每個 Vue 實例在被創建時都要經過一系列的初始化過程——例如,需要設置數據監聽、編譯模板、將實例掛載到 DOM 併在數據變化時更新 DOM 等。同時在這個過程中也會運行一些叫做生命周期鉤子的函數,這給了用戶在不同階段添加自己的代碼的機會。 2、圖示: ...
  • netty服務端啟動 ServerBootstrap源碼解析 前面的第一篇文章中,我以spark中的netty客戶端的創建為切入點,分析了netty的客戶端引導類Bootstrap的參數設置以及啟動過程。顯然,我們還有另一個重要的部分 服務端的初始化和啟動過程沒有探究,所以這一節,我們就來從源碼層面 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...