JS设计模式八:观察者模式

来源:互联网 发布:js插件如何编写 闭包 编辑:程序博客网 时间:2024/05/18 02:46
JS设计模式八:观察者模式2015-02-09 10:42:18

分类: JavaScript


    观察者模式对于前端开发来说也是一种接触较为频繁的软件设计模式,如果没注意过,那说明你还要好好学习啦。
    观察者模式又称做发布订阅模式,基本定义为一对多或者多对多的依赖关系,由一个或者多个观察者对象来对一个或者多个发布者进行监听,由发布者的行为来触发各个观察者的行为。
    观察者模式使的可以支持简单的广播通信,自动通知所有订阅过的对象,从而使得目标对象和各个观察者可以各自独立,减弱耦合度,使两者更容易扩展和重用。
    简单的观察者模式也就无非为一个简单的对象,内部有一个存放函数的数组 list, add以及fire方法,add方法用来添加监听的回调函数进入lists数组,fire则是传递参数触发回调函数列表。这是JS中以回调函数形式实现观察者模式最简单的一种方式。

  1. var Observable= {
  2.     lists :[],
  3.     add: function( fn){
  4.         this.lists.push( fn );
  5.     },
  6.     fire :function( args){
  7.         this.lists.forEach(fucntion(){
  8.             fn( args);
  9.         });
  10.     }
  11. }

    结果当然是biubiu弹出两个console了。下面是一个稍微复杂点的观察者模式,包含了订阅,发布以及退订三个方法,基本可以实现一些简单的观察者模式。以围绕存储回调函数列表的对象 topics 进行订阅,发布和退订。

  1. (function(){

  2.     var q= window.pubsub= {};

  3.     var topics= {},
  4.         subUid = 0;

  5.     // 发布
  6.     q.publish= function (topic, args){

  7.         if(!topics[topic]){
  8.             returnfalse;
  9.         }

  10.         var subscribers= topics[topic],
  11.             len = subscribers? subscribers.length: 0,
  12.             i = 0;

  13.         while(i< len){
  14.             subscribers[i].func(args);
  15.             i++;
  16.         }

  17.         returntrue;
  18.     };

  19.     //订阅方法
  20.     q.subscribe= function (topic, func){
  21.         if(!func){
  22.             returnfalse;
  23.         }
  24.        
  25.         if(!topics[topic]){
  26.             topics[topic]= [];
  27.         }

  28.         var token= subUid++;

  29.         topics[topic].push({
  30.             token: token,
  31.             func: func
  32.         });

  33.         return token;
  34.     };

  35.     //退订方法
  36.     q.unsubscribe= function (topic, token){

  37.         if(!topics[topic]){
  38.             returnfalse;
  39.         }

  40.         var subscribers= topics[topic],
  41.             len = subscribers? subscribers.length: 0,
  42.             i = 0;

  43.         for(; i< len; i++){
  44.             if(subscribers[i].token=== token){
  45.                 subscribers.splice(i, 1);
  46.                 return token;
  47.             }
  48.         }
  49.         returnfalse;
  50.     };
  51. })();

    一般来说这是JS中实现观察者模式最基本的形式,数组来存放回调,手动进行回调函数的添加,删除和触发。有点类似于一些库中的回调函数的形式。在复杂一点就是处理一下对于数据的储存方式等,下面是之前写过的一个自己的观察者模式实现方式:

    subpub 模块,实现发布者与订阅者双向记录绑定 即:发布者中记录订阅该发布者的所有订阅者, 订阅者中记录该订阅者订阅的所有发布者。
    实现创建 publisher 发布者 和 subscriber 订阅者 。
    publisher 实现 1. publish 发布信息 2. addSubscriber 添加订阅者 3. deleteSubscriber 删除订阅者 4. clear 清空订阅者。
    subscriber 实现 1. subscribe 订阅发布者 2. unsubscribe 取消订阅发布者 3. triggle 触发订阅者执行函数


  1. (function( subpub){
  2.     if(typeof define==="function" && define.amd){

  3.         define( subpub);

  4.     }elseif(window.jQuery){

  5.         var old= jQuery.subpub,
  6.             sp = subpub();
  7.         jQuery.subpub= sp;

  8.         jQuery.template.noConflict= function (){
  9.             jQuery.subpub= old;
  10.             return sp;
  11.         }

  12.     }elseif(!window.subpub){
  13.         window.subpub= subpub();
  14.     }else{
  15.         thrownew Error('Variable cookie is already exit in window');
  16.     }
  17. }(function(){
  18.    
  19.     // 由于发布者 订阅者是双向存储标记的 所以使用发布者ID和订阅者ID进行标记
  20.     // 发布和订阅总对象来关联 ID 与 对象
  21.     var PUBID= 0,
  22.         pubObj ={},
  23.         SUBID = 0,
  24.         subObj ={};

  25.     // 关于对象类型的均不采用鸭子辨别法 只用全等
  26.     function has(array,item ){
  27.         varvalue,i,len, hasOwn = {}.hasOwnProperty;
  28.         if(array==null)return;

  29.         if(array instanceof Array ){
  30.             for(i=0, len=array.length; i<len; i++){
  31.                 if(array[i]===item ){
  32.                     returntrue;
  33.                 }
  34.             }
  35.         }else{
  36.             for(varkey in array){
  37.                 if(hasOwn.call(array,key)&&array[key]===item){
  38.                     returntrue;
  39.                 }
  40.             }
  41.         }
  42.         returnfalse;
  43.     }

  44.     function each(array, iterator, context){
  45.         varvalue,i,len,
  46.             forEach =[].forEach,
  47.             hasOwn ={}.hasOwnProperty;

  48.         context = context||this;

  49.         if(array==null)return;

  50.         if(forEach&&array.forEach=== forEach){
  51.             array.forEach(iterator, context);
  52.         }elseif(array instanceof Array ){
  53.             for(i=0, len=array.length; i<len; i++){
  54.                 iterator.call(context,array[i], i, array);
  55.             }
  56.         }else{
  57.             for(varkey in array){
  58.                 if(hasOwn.call(array,key)){
  59.                     iterator.call(context,array[key],key,array);
  60.                 }
  61.             }
  62.         }
  63.     }

  64.     function publisher(){

  65.         // 本来想将该属性私有封装,不过代码量会增加很多,并且也没有什么必要
  66.         this.subIds= [];
  67.         this.pubID= PUBID;
  68.     }

  69.     publisher.prototype= {

  70.         // 发布 触发所有的 subIds id 列表成员的 trigger
  71.         publish :function( info){
  72.             var subIds= this.subIds;

  73.             each( subIds,function( subId){
  74.                 subObj[subId].trigger( info );
  75.             });

  76.             returnthis;
  77.         },

  78.         // 发布者中添加 订阅者, 并在订阅者中记录发布者 单一订阅原则
  79.         addSubscriber :function( subscriber){
  80.             var subIds= this.subIds,
  81.                 subId = subscriber.subID;

  82.             // 单一订阅 也可以避免 订阅者 发布者中相互记录的无限循环
  83.             if( has( subIds, subId) ){
  84.                 return;
  85.             }

  86.             if(subscriber&& subscriber.subscribe){
  87.                 subIds.push( subId);
  88.                 subscriber.subscribe(this );
  89.             }else{
  90.                 thrownew Error('subscriber is illegal');
  91.             }
  92.         },

  93.         deleteSubscriber :function( subscriber){
  94.             var subIds= this.subIds,
  95.                 subId = subscriber.subID;

  96.             if(!has( subIds, subId ) ){
  97.                 return;
  98.             }

  99.             for(var i= subIds.length- 1; i>= 0; i--){
  100.                 if( subIds[i]=== subId){
  101.                     subIds.splice(i, 1);
  102.                 }
  103.             };

  104.             subscriber.unsubscribe(this );

  105.             returnthis;
  106.         },

  107.         clear: function(){
  108.             var subIds= this.subIds;
  109.             for(var i= subIds.length- 1; i>= 0; i--){
  110.                 subObj[ subIds[i]].unsubscribe(this );
  111.             };

  112.             this.subIds= [];
  113.         }
  114.     }


  115.     function subscriber( func){
  116.         this.pubIds= [];
  117.         this.subID= SUBID;
  118.         this.callbck= func;
  119.     }

  120.     subscriber.prototype= {
  121.         subscribe :function( publisher){
  122.             var pubIds= this.pubIds,
  123.                 pubId = publisher.pubID;

  124.             if( has( pubIds, pubId) ){
  125.                 return;
  126.             }

  127.             if( publisher&& publisher.addSubscriber){
  128.                 pubIds.push( pubId);
  129.                 publisher.addSubscriber(this);
  130.             }else{
  131.                 thrownew Error('publisher is illegal');
  132.             }
  133.         },

  134.         unsubscribe :function( publisher){
  135.             var pubIds= this.pubIds,
  136.                 pubId = publisher.pubID;

  137.             if(!has( pubIds, pubId ) ){
  138.                 return;
  139.             }

  140.             for(var i= pubIds.length- 1; i>= 0; i--){
  141.                 if( pubIds[i]=== pubId){
  142.                     pubIds.splice(i, 1);
  143.                 }
  144.             };

  145.             publisher.deleteSubscriber(this );

  146.             returnthis;
  147.         },

  148.         clear: function(){
  149.             var pubIds= this.pubIds;
  150.             for(var i= pubIds.length- 1; i>= 0; i--){
  151.                 pubObj[ pubIds[i]].deleteSubscriber(this );
  152.             };

  153.             this.pubIds= [];
  154.         },

  155.         trigger :function( args){
  156.             this.callbck( args );
  157.         }
  158.     }

  159.     return{
  160.         creatPublisher :function(){
  161.             var pub= new publisher();
  162.             pubObj[ PUBID++] = pub;

  163.             return pub;
  164.         },
  165.         creatSubscriber :function( func){
  166.             varsub = new subscriber( func);
  167.             subObj[ SUBID++] = sub;

  168.             returnsub;
  169.         }
  170.     }
  171. }));

    实例:

  1. var sp= window.subpub;

  2. var a1= sp.creatPublisher(), a2 = sp.creatPublisher(), a3 = sp.creatPublisher();
  3. var b1= sp.creatSubscriber(function(a){ console.log('b1 '+a)}),
  4.     b2 = sp.creatSubscriber(function(a){ console.log('b2 '+a)}),
  5.     b3 = sp.creatSubscriber(function(a){ console.log('b3 '+a)});

  6. a1.addSubscriber(b1)
  7. a1.addSubscriber(b3)
  8. a2.addSubscriber(b2)
  9. a3.addSubscriber(b1)
  10. a3.addSubscriber(b2)
  11. a3.addSubscriber(b3)


  12. < a1.publish("this is a1 publish")
  13. > "b1 this is a1 publish"
  14. > "b3 this is a1 publish"

  15. < a3.deleteSubscriber(b1)
  16. < a3.deleteSubscriber(b2)
  17. < a3.publish("this is a3 publish")
  18. > "b3this is a3 publish"

  19. < b3.unsubscribe(a3)
  20. < b2.subscribe(a3)
  21. < a3.publish("this is a3 publish")
  22. > "b2 this is a3 publish"

    简单学习了那么些设计模式之后,其实越来越觉得设计模式开始越来越抽象化,成为一种类似于现实如何转化为逻辑,思维的一种方式。所以个人认为,如何将事物抽象化,各种方式主要解决的针对性问题是什么,或许就是设计模式存在的真正意义吧。

    不足之处,望读者指出,谢谢~
0 0
原创粉丝点击