觀察者模式和發佈訂閱模式(下)

来源:https://www.cnblogs.com/liuyongjia/archive/2018/08/03/9415008.html
-Advertisement-
Play Games

發佈訂閱模式 前一篇對觀察者模式做了介紹,重點在於觀察者和被觀察者的對應關係,以及將被觀察者的改變及時通知到相對應的觀察者。 這樣的模式基本上可以解決少量數據源的情景,在觀察者和被觀察者可能是多對多關係的情況下,強耦合的結構會讓代碼不夠清晰,難以維護。 在《JavaScript設計模式》一書中,提到 ...


發佈訂閱模式

前一篇對觀察者模式做了介紹,重點在於觀察者和被觀察者的對應關係,以及將被觀察者的改變及時通知到相對應的觀察者。
這樣的模式基本上可以解決少量數據源的情景,在觀察者和被觀察者可能是多對多關係的情況下,強耦合的結構會讓代碼不夠清晰,難以維護。
在《JavaScript設計模式》一書中,提到了Observer和Publish/Subscribe的區別。

Observer模式要求希望接收到主題同志的觀察者(或對象)必須訂閱內容改變的事件。
Publish/Subscribe模式使用了一個主題/事件通道,這個通道介於希望接收到通知(訂閱者)的對象和激活事件的對象(發佈者)之間。該事件系統允許代碼定義應用程式的特定事件,這些事件可以傳遞自定義參數,自定義參數包含訂閱者所需的值。其目的是避免訂閱者和發佈者之間產生依賴關係。

這裡的關鍵點在於,通過一個事件中心,將發佈者和訂閱者的耦合關係解開,發佈者和訂閱者通過事件中心來產生聯繫。
打個比方,發佈者像是發佈小廣告的,事件中心是一個調度站,訂閱者則告訴事件中心,我關註A、B類型的廣告,如果有更新,請通知我。調度站記錄A,B類型下的訂閱者,等到A,B廣告發佈時,通知到訂閱者。
這個例子里,發佈者不關心訂閱者是誰,也不維護訂閱者列表,同訂閱者解耦,只將自己發佈的內容提交到事件中心。而訂閱者和主題的關係,交給了事件中心來維護。
畫一個類圖來解釋一下他們的關係。
發佈訂閱模式類圖
先來定義兩個虛類:Publisher 和 Subscriber。

abstract class Publisher {
    data: string;
    id: string;
    abstract publish(any);
}

abstract class Subscriber {
    id: string;
    abstract subscribe(topicId: string);
    abstract update(topicData: string);
}

接著來繼承這兩個類,聲明兩個實體類:

class ApplePublisher extends Publisher {
    private _data;
    private _id;
    private channel;
    constructor (defaultId, defaultData, defaultChannel: TopicChannel) {
        super();
        this._id = defaultId;
        this._data = defaultData;
        this.channel = defaultChannel;
    }

    get id () {
        return this._id;
    }

    set id (newId) {
        this._id = newId;
    }

    get data () {
        return this._data;
    }

    set data (newData) {
        this._data = newData;
    }

    publish () {
        this.channel.publishBaseId(this.id);
    }
}

class FruitSubscriber extends Subscriber {

    readonly _id;
    readonly _publishId;
    private channel;

    constructor (id: string, publishId: string, topicChannel: TopicChannel) {
        super();
        this._id = id;
        this._publishId = publishId;
        this.channel = topicChannel;

        this.subscribe(this._publishId);
    }

    subscribe (topicId: string) {
        this.channel.subscribe(topicId, this);
    }

    update (topicData: string) {
        console.log('fruit subscriber ' + this._id + ': ' + topicData);
    }
}

最後來實現TopicChannel,這個類用來處理訂閱、發佈通知功能。

class TopicChannel {
    private publisherMap;
    private subscriberMap;

    constructor () {
        this.publisherMap = new Map<string, Publisher>();
        this.subscriberMap = new Map<string, Array<Subscriber>>();
    }

    addPublisher (publisher: Publisher) {
        this.publisherMap.set(publisher.id, publisher);
    }

    removePublisher (publisher: Publisher) {
        this.publisherMap.delete(publisher.id)
    }

    clearPublisher () {
        this.publisherMap.clear();
    }

    subscribe (publisherId: string, subscriber: Subscriber) {
        if (this.subscriberMap.has(publisherId)) {
            this.subscriberMap.get(publisherId).push(subscriber);
        } else {
            this.subscriberMap.set(publisherId, [subscriber]);
        }
    }

    publishBaseId (publisherId: string) {
        if (this.publisherMap.has(publisherId)) {
            this.subscriberMap.get(publisherId).forEach((item)=>{
                item.update(this.publisherMap.get(publisherId).data);
            })
        } else {
            console.log('There is not the publisher!');
        }
    }

}

TopicChannel通過TopicId來維護髮布者和訂閱者的關係,使得發佈者和訂閱者充分解耦。
使得訂閱者可以訂閱多個主題,在內部根據主題的不同,執行不同的邏輯。
發佈者則完全無視訂閱者的邏輯,只管將自己的內容推送到TopicChannel。

let topicChannel = new TopicChannel();

let applePublisher1 = new ApplePublisher('apple publisher1', 'foo apple1', topicChannel);
let applePublisher2 = new ApplePublisher('apple publisher2', 'foo apple2', topicChannel);

topicChannel.addPublisher(applePublisher1);
topicChannel.addPublisher(applePublisher2);

let fruitSubscriber1 = new FruitSubscriber('fruit1', 'apple publisher1', topicChannel);
let fruitSubscriber2 = new FruitSubscriber('fruit2', 'apple publisher2', topicChannel);
let fruitSubscriber3 = new FruitSubscriber('fruit3', 'apple publisher1', topicChannel);

fruitSubscriber2.subscribe('apple publisher1');

applePublisher1.publish();
applePublisher2.publish();

結果為:

fruit subscriber fruit1: foo apple1
fruit subscriber fruit3: foo apple1
fruit subscriber fruit2: foo apple1
fruit subscriber fruit2: foo apple2

發佈訂閱模式將發佈者和訂閱者完全解耦,由事件中心通過topicId或者是其他唯一key來維護二者的關係,使得程式可以分割成更小,內聚更高的模塊。
與此同時,由於弱化了發佈者與訂閱者的關係,使得發佈者難以追蹤到訂閱者,無法獲得來自訂閱者反饋;並且同一主題的訂閱者之間相對透明,不能產生聯動。
以上,如有錯誤,敬請指正,感謝閱讀。


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

-Advertisement-
Play Games
更多相關文章
  • GsonFormat插件用於在androidStudio 根據json自動生成class的欄位和方法,極大提高了開發效率 一、安裝GsonFormat插件 二、重啟Android Studio,新建一個java類 ,然後在文件內 按快捷鍵alt+Insert 彈出菜單選擇GsonFormat ,複製 ...
  • 開頭先寫一句理論:所謂狀態機,是一種抽象的數據模型,是“事物發展的趨勢”,其原理是事件驅動。廣泛地講,世界萬物都是狀態機。 一、狀態機是一種抽象的數據模型 在react中,props和state都可以用來傳遞數據。這裡作一下區分。 1.props props用於組件間的數據傳遞。其本身只是一個屬性, ...
  • (1)HTML5現在已經不是SGML的子集,主要是關於圖像,位置,存儲,地理定位等功能的增加。 繪畫canvas元素; 用於媒介回放的video和audio元素; 本地離線存儲localStorage長期存儲數據,瀏覽器關閉後數據不丟失,sessionStorage的數據在瀏覽器關閉後自動刪除; 語 ...
  • 筆者最近在寫node.js項目中遇到了一個比較有趣的流程式控制制需求需求是:先將多個object對象迴圈遍歷後進行非同步操作寫入到資料庫中去,等這部操作結束後在進行下一次的查詢操作。這個流程需要註意的是在你foreach 或者 for迴圈的時候裡面的非同步操作是不會同步進行的,往往會導致時間相差而產生bug ...
  • 首先解釋下宿主環境:一般宿主環境由外殼程式創建與維護,只要能提供js引擎執行的環境都可稱之為外殼程式。如:web瀏覽器,一些桌面應用系統等。即由web瀏覽器或是這些桌面應用系統早就的環境即宿主環境。 1、本地對象 ECMA-262 把本地對象(native object)定義為“獨立於宿主環境的 E ...
  • 新手一枚,解決的問題喜歡記錄,也許正好有人在網上迷茫的百度著。-0- 最近使用Chart.js做折線圖的報表展示,直接顯示整數啥的很好弄畢竟例子直接在哪裡可以用,百分比就沒辦法了。百度慢慢汲取營養,雖然總是幾篇文章複製粘貼,但還有有收穫,然後自己搗鼓半天總算是弄出來了。。。 首先參考: http:/ ...
  • /** * @param {number[]} height * @return {number} */ var maxArea = function(height) { let maxarea = 0 for(let i=0;i=0; j--){ maxarea = Math.max(maxare... ...
  • 學習CSS相關知識,定位是其中的重點,也是難點之一,如果不瞭解css定位有時候都不知道怎麼用,下麵整理了一下關於定位屬性的具體理解和應用方案。 一:定位 定位屬性列表 position top bottom right left z index position 基本語法: position:sta ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...