[JS設計模式]:觀察者模式(即發佈-訂閱者模式)(4)

来源:https://www.cnblogs.com/moqiutao/archive/2019/07/09/11139165.html
-Advertisement-
Play Games

簡介 觀察者模式又叫發佈 訂閱模式,它定義了對象間的一種一對多的關係,讓多個觀察者對象同時監聽某一個主題對象,當一個對象發生改變時,所有依賴於它的對象都將得到通知。 舉一個現實生活中的例子,例如小紅在淘寶的一家店裡看上了一雙紅色的鞋,小李也在這家店裡面看上了一頂黑色的帽子,但是聯繫賣家時,賣家回答這 ...


簡介

觀察者模式又叫發佈---訂閱模式,它定義了對象間的一種一對多的關係,讓多個觀察者對象同時監聽某一個主題對象,當一個對象發生改變時,所有依賴於它的對象都將得到通知。

舉一個現實生活中的例子,例如小紅在淘寶的一家店裡看上了一雙紅色的鞋,小李也在這家店裡面看上了一頂黑色的帽子,但是聯繫賣家時,賣家回答這兩樣都沒貨了。賣家告訴小紅小李,要是喜歡的話,可以關註下店鋪,到貨了,我會給大家通知的。這就是一個典型的發佈-訂閱模式,賣家是發佈者,買家是訂閱者。當貨來的時候,會依次通知小紅、小李等買家,依次給他們發消息通知。

代碼實現

var goodsObj = {};
goodsObj.list = []; //存放訂閱者

//訂閱者
goodsObj.subscribe = function(key,fn){
    if(!this.list[key]){
        this.list[key] = [];
    }
    //訂閱的消息存放到緩存列表中
    this.list[key].push(fn);
}

//取消訂閱
goodsObj.unsubscribe = function(key,fn){
    var fns = this.list[key];
    //如果key對應的消息沒有訂閱過的話,則返回
    if(!fns){ 
        return false;
    }
    //如果沒有傳入具體的回調函數,表示取消key對應消息的所有訂閱
    if(!fn){
        fns && (fns.length = 0);
    }else{
        for(var i = fns.length - 1; i >= 0; i--){
            if(fns[i] === fn){
                //刪除訂閱者的回調函數
                fns.splice(i,1); 
            }
        }
    }
}

//發佈者
goodsObj.publish = function(){
    var key = Array.prototype.shift.call(arguments);
    var fns = this.list[key];
    //如果沒有訂閱過該key的消息,直接返回
    if(!fns || fns.length === 0){
        return false;
    }
    for(var i = 0,fn; fn = fns[i++];){
        fn.apply(this,arguments);
    }
}

//小紅訂閱了鞋
goodsObj.subscribe('shoes',fn1 = function(color){
    console.log('訂閱鞋的顏色:' + color + '到貨了,可以拍了');
});

//小李訂閱了帽子
goodsObj.subscribe('cap',fn2 = function(size){
    console.log('訂閱帽子的尺寸:' + size  + '到貨了,可以拍了');
});

//發佈
goodsObj.publish('shoes','red');
goodsObj.publish('cap',40);

//取消訂閱
goodsObj.unsubscribe('shoes',fn1);

//發佈消息,看是否能收到
goodsObj.publish('shoes','block');

 優化取消訂閱

為了優化取消訂閱,在訂閱的時候,給每個訂閱者一個不同的標識,代碼如下:

var goodsObj = {};
goodsObj.list = {}; //存放訂閱者
var subuid = -1; //每個訂閱者標識不一樣

//訂閱者
goodsObj.subscribe = function(key,fn){
    if(!this.list[key]){
        this.list[key] = [];
    }
    //訂閱的消息存放到緩存列表中
    var token= (++subuid).toString();
    this.list[key].push({
        token:token,
        fn:fn
    });
    return token;
}

//取消訂閱
goodsObj.unsubscribe = function(token){
    for(var m in goodsObj.list){
        if(goodsObj.list.hasOwnProperty(m)){
            for(var i = 0, j = goodsObj.list[m].length; i < j; i++ ){
                if(goodsObj.list[m][i].token == token){
                    goodsObj.list[m].splice(i,1);
                    break;
                }
            }
        }
    }
}

//發佈者
goodsObj.publish = function(){
    var key = Array.prototype.shift.call(arguments);
    var fns = this.list[key];
    //如果沒有訂閱過該key的消息,直接返回
    if(!fns || fns.length === 0){
        return false;
    }
    for(var i = 0,obj; obj = fns[i++];){
        obj.fn.apply(this,arguments);
    }
}

//小紅訂閱了鞋
var token1 = goodsObj.subscribe('shoes',function(color){
    console.log('訂閱鞋的顏色:' + color + '到貨了,可以拍了');
});

//小李訂閱了帽子
goodsObj.subscribe('cap',function(size){
    console.log('訂閱帽子的尺寸:' + size  + '到貨了,可以拍了');
});

//發佈
goodsObj.publish('shoes','red');
goodsObj.publish('cap',40);

//取消訂閱
goodsObj.unsubscribe(token1);

//發佈消息,看是否能收到
goodsObj.publish('shoes','block');

封裝全局發佈訂閱對象

封裝一個全局的發佈-訂閱對象,代碼如下:

var Event = (function(){
    var list = {}, //存放訂閱者
        subscribe,
        unsubscribe,
        publish,
        subuid = -1; //每個訂閱者標識不一樣
        
        //訂閱者
        subscribe = function(key,fn){
            if(!list[key]){
                list[key] = [];
            }
            //訂閱的消息存放到緩存列表中
            var token= (++subuid).toString();
            list[key].push({
                token:token,
                fn:fn
            });
            return token;
        };
        
        //取消訂閱
        unsubscribe = function(token){
            for(var m in list){
                if(list.hasOwnProperty(m)){
                    for(var i = 0, j = list[m].length; i < j; i++ ){
                        if(list[m][i].token == token){
                            list[m].splice(i,1);
                            break;
                        }
                    }
                }
            }
        };
        
        //發佈者
        publish = function(){
            var key = Array.prototype.shift.call(arguments);
            var fns = list[key];
            //如果沒有訂閱過該key的消息,直接返回
            if(!fns || fns.length === 0){
                return false;
            }
            for(var i = 0,obj; obj = fns[i++];){
                obj.fn.apply(this,arguments);
            }
        };
        
        return {
            publish:publish,
            subscribe:subscribe,
            unsubscribe:unsubscribe
        }        
})();

//小紅訂閱了鞋
var token1 = Event.subscribe('shoes',function(color){
    console.log('訂閱鞋的顏色:' + color + '到貨了,可以拍了');
});

//小李訂閱了帽子
Event.subscribe('cap',function(size){
    console.log('訂閱帽子的尺寸:' + size  + '到貨了,可以拍了');
});

//發佈
Event.publish('shoes','red');
Event.publish('cap',40);

//取消訂閱
Event.unsubscribe(token1);

//發佈消息,看是否能收到
Event.publish('shoes','block');

實例

我們使用上面封裝的全局的發佈-訂閱對象來實現兩個模塊之間的通信問題;比如現在有一個頁面有一個按鈕,每次點擊此按鈕後,div中會顯示此按鈕被點擊的總次數;如下代碼:

<button id="clickBtn">點擊我計數</button>
<div id="showCount">0</div>

JS代碼如下:

(function(){
    var count = 0;
    //發佈(責處理點擊操作、發佈消息)
    document.getElementById('clickBtn').addEventListener('click',function(){
        Event.publish('add',count++);
    },false);
    
    //訂閱(負責監聽add這個消息,並把點擊的總次數顯示到頁面上來)
    Event.subscribe('add',function(curCount){
        document.getElementById('showCount').innerHTML = curCount;
    })
})();

訂閱發佈更精簡代碼:

var Event = function(){
    var listen,obj,remove,one,trigger,__this;
    obj = {};
    __this = this;
    listen = function(key,eventfn){
        var _ref,stack;
        stack = (_ref = obj[key]) != null ? _ref : obj[key] = [];
        return stack.push(eventfn);
    };
    one = function(key,eventfn){
        remove(key);
        return listen(key,eventfn);
    };
    remove = function(key){
        var _ref;
        return (_ref = obj[key]) != null ? _ref.length = 0 : void 0;
    };
    trigger = function(){
        var fn,stack,_i,_len,_ref,key;
        key = Array.prototype.shift.call(arguments);
        stack = (_ref = obj[key]) != null ? _ref :obj[key] = [];
        for(_i=0,_len=stack.length;_i<_len;_i++){
            fn = stack[_i];
            if(fn.apply(__this,arguments) === false){
                return false;
            }
        }
    }
    
    return {
        listen:listen,
        one:one,
        remove:remove,
        trigger:trigger
    }
}

var addTv = Event();

addTv.listen('play',function(data){
    console.log('今天上午要播放的電影是:' +data.name);
});
addTv.one('play',function(data){
    console.log('只有我訂閱的電影信息:' +data.name);
});
addTv.trigger('play',{'name':'國產凌凌漆'});

addTv.remove('play');
addTv.trigger('play',{'name':'國產凌凌漆22'});

 參考


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

-Advertisement-
Play Games
更多相關文章
  • 解釋CSS3 中新增的選擇器中最具有代表性的就是序選擇器,大致可以分為兩類: (1)同級別的第幾個(2)同類型的第幾個 先寫一個公共代碼 1.選中同級別中的第一個 註意點:不區分類型,只管取第一個,不管第一個是什麼標簽 解釋:在同級別中只選取第一個為h1標簽和div下的p標簽,然後在這些裡面只選p標 ...
  • Javascript是前端面試的重點,本文重點梳理下 Javascript 中的常考基礎知識點,然後就一些容易出現的題目進行解析。限於文章的篇幅,無法將知識點講解的面面俱到,本文只羅列了一些重難點。 ...
  • Vue.js提供了v-model指令用於雙向數據綁定,比如在輸入框上使用時,輸入的內容會事實映射到綁定的數據上,綁定的數據又可以顯示在頁面里,數據顯示的過程是自動完成的。 v-model本質上不過是語法糖。它負責監聽用戶的輸入事件以更新數據,並對一些極端場景進行一些特殊處理。例如: 渲染如下: 當我 ...
  • 背景 在Electron打開新視窗的時候,提前載入一段JavaScript腳本,以此內置一些屬性或介面給被打開的頁面。之所以要以註入方式,而不是頁面自己引用,原因是不想麻煩頁面自行引用,不想修改舊有的業務邏輯。 方法一 一開始是想在打開BrowserWindow後,執行executeJavaScri ...
  • 先鋪墊下原型規則: 1.所有的引用類型(數組,對象,函數)都具有對象特性,可自由擴展屬性(出了null外) 2.所有的引用類型(數組,對象,函數)都有一個__proto__屬性(隱式原型),屬性值是一個對象 3.所有的函數都有一個prototype屬性(顯示原型),屬性值是一個對象 4.所有的引用類 ...
  • 好的代碼像粥一樣,都是用時間熬出來的。 概述 文件 I/O 是由簡單封裝的標準 POSIX 函數提供的。 通過 require('fs') 使用該模塊。 所有文件系統操作都具有同步和非同步的形式。 非同步的形式總是將完成回調作為其最後一個參數。 傳給完成回調的參數取決於具體方法,但第一個參數始終預留用於 ...
  • 問題現象 這個問題的現象說起來很簡單。 小程式頁面中有一篇很長的文章,內部有一個Echarts圖表,手指上下滑動觀看內容。 但是手指滑動區域在Echarts圖表上時,頁面卻不能滑動了。 如下圖: 追蹤問題原因 因為在小程式上渲染圖表用到的是 "echarts for weixin" 這個組件,而這個 ...
  • 編程式跳轉 router.js: ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...