国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

JavaScript編程設計模式之觀察者模式(Observer Pattern)實例詳解

2019-11-19 15:05:12
字體:
來源:轉載
供稿:網友

本文實例講述了JavaScript編程設計模式之觀察者模式。分享給大家供大家參考,具體如下:

簡介

簡單的解釋觀察者模式,就是一個對象(subject)維護一個依賴他的對象(observers)列表,當自身狀態發生變化時,自動通知所有觀察者對象。當某個對象不需要獲得通知時,可以從對象列表中刪除掉。

從上面的解釋中我們可以提煉出三個componet: Subject, ObserverList和Observer,用JS實現很簡單:

function ObserverList(){ this.observerList = [];}ObserverList.prototype.Add = function( obj ){ return this.observerList.push( obj );};ObserverList.prototype.Empty = function(){ this.observerList = [];};ObserverList.prototype.Count = function(){ return this.observerList.length;};ObserverList.prototype.Get = function( index ){ if( index > -1 && index < this.observerList.length ){  return this.observerList[ index ]; }};ObserverList.prototype.Insert = function( obj, index ){ var pointer = -1; if( index === 0 ){  this.observerList.unshift( obj );  pointer = index; }else if( index === this.observerList.length ){  this.observerList.push( obj );  pointer = index; } return pointer;};ObserverList.prototype.IndexOf = function( obj, startIndex ){ var i = startIndex, pointer = -1; while( i < this.observerList.length ){  if( this.observerList[i] === obj ){   pointer = i;  }  i++; } return pointer;};ObserverList.prototype.RemoveAt = function( index ){ if( index === 0 ){  this.observerList.shift(); }else if( index === this.observerList.length -1 ){  this.observerList.pop(); }};// Extend an object with an extensionfunction extend( extension, obj ){ for ( var key in extension ){  obj[key] = extension[key]; }}

Subject擁有增加和刪除Observer的能力

function Subject(){ this.observers = new ObserverList();}Subject.prototype.AddObserver = function( observer ){ this.observers.Add( observer );};Subject.prototype.RemoveObserver = function( observer ){ this.observers.RemoveAt( this.observers.IndexOf( observer, 0 ) );};Subject.prototype.Notify = function( context ){ var observerCount = this.observers.Count(); for(var i=0; i < observerCount; i++){  this.observers.Get(i).Update( context ); }};

最后定義一個觀察者對象,實現update方法

// The Observerfunction Observer(){ this.Update = function(){  // ... };}

當有多個觀察者,只需擴展上面的基本對象,并重寫Update方法。

盡管觀察則模式被廣泛使用,但在JS中經常使用它的變體: 發布訂閱模式

發布訂閱模式通過一個topic/event通道,解耦了觀察者模式中Subject(發布者)和Observer(訂閱者)之間耦合的問題,在JS中被廣泛使用。

下面簡單的例子說明了使用發布訂閱模式的基本結構

// A very simple new mail handler// A count of the number of messages receivedvar mailCounter = 0;// Initialize subscribers that will listen out for a topic// with the name "inbox/newMessage".// Render a preview of new messagesvar subscriber1 = subscribe( "inbox/newMessage", function( topic, data ) { // Log the topic for debugging purposes console.log( "A new message was received: ", topic ); // Use the data that was passed from our subject // to display a message preview to the user $( ".messageSender" ).html( data.sender ); $( ".messagePreview" ).html( data.body );});// Here's another subscriber using the same data to perform// a different task.// Update the counter displaying the number of new// messages received via the publishervar subscriber2 = subscribe( "inbox/newMessage", function( topic, data ) { $('.newMessageCounter').html( mailCounter++ );});publish( "inbox/newMessage", [{ sender:"hello@google.com", body: "Hey there! How are you doing today?"}]);// We could then at a later point unsubscribe our subscribers// from receiving any new topic notifications as follows:// unsubscribe( subscriber1, );// unsubscribe( subscriber2 );

發布訂閱模式的實現

許多Js庫都很好的實現了發布訂閱模式,例如Jquery的自定義事件功能。

// Publish// jQuery: $(obj).trigger("channel", [arg1, arg2, arg3]);$( el ).trigger( "/login", [{username:"test", userData:"test"}] );// Dojo: dojo.publish("channel", [arg1, arg2, arg3] );dojo.publish( "/login", [{username:"test", userData:"test"}] );// YUI: el.publish("channel", [arg1, arg2, arg3]);el.publish( "/login", {username:"test", userData:"test"} );// Subscribe// jQuery: $(obj).on( "channel", [data], fn );$( el ).on( "/login", function( event ){...} );// Dojo: dojo.subscribe( "channel", fn);var handle = dojo.subscribe( "/login", function(data){..} );// YUI: el.on("channel", handler);el.on( "/login", function( data ){...} );// Unsubscribe// jQuery: $(obj).off( "channel" );$( el ).off( "/login" );// Dojo: dojo.unsubscribe( handle );dojo.unsubscribe( handle );// YUI: el.detach("channel");el.detach( "/login" );

簡單實現

var pubsub = {};(function(q) {  var topics = {},    subUid = -1;  // Publish or broadcast events of interest  // with a specific topic name and arguments  // such as the data to pass along  q.publish = function( topic, args ) {    if ( !topics[topic] ) {      return false;    }    var subscribers = topics[topic],      len = subscribers ? subscribers.length : 0;    while (len--) {      subscribers[len].func( topic, args );    }    return this;  };  // Subscribe to events of interest  // with a specific topic name and a  // callback function, to be executed  // when the topic/event is observed  q.subscribe = function( topic, func ) {    if (!topics[topic]) {      topics[topic] = [];    }    var token = ( ++subUid ).toString();    topics[topic].push({      token: token,      func: func    });    return token;  };  // Unsubscribe from a specific  // topic, based on a tokenized reference  // to the subscription  q.unsubscribe = function( token ) {    for ( var m in topics ) {      if ( topics[m] ) {        for ( var i = 0, j = topics[m].length; i < j; i++ ) {          if ( topics[m][i].token === token) {            topics[m].splice( i, 1 );            return token;          }        }      }    }    return this;  };}( pubsub ));

使用方法

// Another simple message handler// A simple message logger that logs any topics and data received through our// subscribervar messageLogger = function ( topics, data ) {  console.log( "Logging: " + topics + ": " + data );};// Subscribers listen for topics they have subscribed to and// invoke a callback function (e.g messageLogger) once a new// notification is broadcast on that topicvar subscription = pubsub.subscribe( "inbox/newMessage", messageLogger );// Publishers are in charge of publishing topics or notifications of// interest to the application. e.g:pubsub.publish( "inbox/newMessage", "hello world!" );// orpubsub.publish( "inbox/newMessage", ["test", "a", "b", "c"] );// orpubsub.publish( "inbox/newMessage", { sender: "hello@google.com", body: "Hey again!"});// We cab also unsubscribe if we no longer wish for our subscribers// to be notified// pubsub.unsubscribe( subscription );// Once unsubscribed, this for example won't result in our// messageLogger being executed as the subscriber is// no longer listeningpubsub.publish( "inbox/newMessage", "Hello! are you still there?" );

更多關于JavaScript相關內容可查看本站專題:《javascript面向對象入門教程》、《JavaScript切換特效與技巧總結》、《JavaScript查找算法技巧總結》、《JavaScript錯誤與調試技巧總結》、《JavaScript數據結構與算法技巧總結》、《JavaScript遍歷算法與技巧總結》及《JavaScript數學運算用法總結

希望本文所述對大家JavaScript程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 广宗县| 定南县| 六枝特区| 贵港市| 西乌珠穆沁旗| 昔阳县| 堆龙德庆县| 汉源县| 博爱县| 迁西县| 天长市| 楚雄市| 搜索| 吴桥县| 中方县| 湄潭县| 金阳县| 余干县| 新乡县| 土默特左旗| 灵寿县| 逊克县| 鄂伦春自治旗| 梅河口市| 牙克石市| 江孜县| 临汾市| 林州市| 日照市| 康定县| 焦作市| 海城市| 碌曲县| 贡嘎县| 宁国市| 宁晋县| 敖汉旗| 德兴市| 滨州市| 荥阳市| 丹棱县|