PubSub的一种实现
今天在浏览JavaScript事件时,复习了下Dean Edward大神的addEvent。突然觉得可以基于他的思路实现一个结构更好的PubSub。
思路也很简单,就是要维护一个类似如下的一个仓库结构:
/*
{
'sayHello': {
0: fn0,
1: fn1,
//... },
'sayGoodBye': {
0: fnn,
//...
},
//...
}*/
下面是我的实现代码:
(function(exports) {
var PubSub = exports.PubSub || {};
//在PubSub对象上增加静态域PubSubCache,用于保存subscribe相关数据
/**
* PubSub.PubSubCache仓库结构
* {
* 'sayHello': {
* 0: fn0,
* 1:fn1,
* //..。
* },
* 'sayGoodBye': {
* //...
* }
* }
*
*/
PubSub.PubSubCache = PubSub.PubSubCache || {$uid: 0};
//PubSub有4个静态方法:subscribe, subscribeOne, unsubscribe, publish
//PubSub不会与DOM元素有关系。这样publish也只能手动去触发了
PubSub.subscribe = function(type, handler) {
var cache = this.PubSubCache[type] || (this.PubSubCache[type] = {});
handler.$uid = handler.$uid || this.PubSubCache.$uid++;
//把回调放入仓库中
cache[handler.$uid] = handler;
};
PubSub.unsubscribe = function(type, handler) {
var counter = 0,$type, cache = this.PubSubCache[type];
if(arguments.length === 1) {
//直接删除此种类型的订阅对象
if(!cache) return true;
return !!this.PubSubCache[type] && (delete this.PubSubCache[type]);
} else if(arguments.length === 2) {
!!this.PubSubCache[type] && (delete this.PubSubCache[type][handler.$uid]);
}
//PubSubCahe仓库中某类型订阅为空,则要删除这个订阅对象
for($type in cache) {
counter++;
}
return !counter && (delete this.PubSubCache[type]);
};
PubSub.publish = function(type) {
var cache = this.PubSubCache[type], key, oneFlag, tmp, context, args = [].slice.call(arguments);
if(!cache) return;
if(args.length === 1) {
context = exports;
} else {
context = args[1];
}
//执行回调
for(key in cache) {
tmp = cache[key];
//在发布消息时可以指定回调函数的上下文,同时还可以传入参数
cache[key].apply(context, args.slice(1));
tmp.one && this.unsubscribe(type, tmp);
}
};
PubSub.subscribeOne = function(type, handler) {
this.subscribe(type, handler);
//给函数加一个只执行一次的标志
handler.one = true;
};
exports.PubSub = PubSub;
})(window);
下面是测试代码:
var data = {name: 'haha', age:18};
var handler2 = function(data) {
console.log('say.hello excuted! 2');
console.log(this.name);
};
//订阅say.hello消息
PubSub.subscribe('say.hello', function(data) {
console.log('say.hello excuted! 1');
});
//第二次订阅say.hello消息
PubSub.subscribe('say.hello', handler2);
//第三次订阅say.hello消息
PubSub.subscribe('say.hello', function(data) {
console.log('say.hello excuted! 3');
console.log(this.age);
});
//第四次增加一个只会执行一次的say.hello消息订阅
PubSub.subscribeOne('say.hello', function(data) {
console.log('say.hello excuted! one');
});
/**
* 发布say.hello消息类型
* 输出:
* say.hello excuted! 1
* say.hello excuted! 2
* haha
* say.hello excuted! 3
* 18
* say.hello excuted! one
*/
PubSub.publish('say.hello', data);
//取消第二次订阅的say.hello类型
PubSub.unsubscribe('say.hello', handler2);
/**
* 发布say.hello消息类型
* 输出:
* say.hello excuted! 1
* say.hello excuted! 3
* 18
*/
console.log('--------------------------------')
PubSub.publish('say.hello', data);
/**
* 再次发布say.hello消息,不过这次除了传入执行上下文外,还要传入参数
* 输出:
* say.hello excuted! 1
* say.hello excuted! 3
* 18
* say.hello excuted! has deliverd args
* args123
*/
PubSub.subscribe('say.hello', function(data, args) {
console.log('say.hello excuted! has deliverd args');
console.log(args);
});
console.log('--------------------------------')
PubSub.publish('say.hello', data, 'args123');
小结:
全局的PubSub对象有四个方法:
1. subscribe(type, handler) :增加一种类型的消息订阅。类似jQuery的bind();
2. subscribeOne(type, handler):增加一种回调只会执行一次的消息订阅,类似jQuery的one()
3. unsubscribe(type, [handler]): 如果只传type,会删除所有的type消息订阅;传入了回调函数,则只删除那一个回调。类似jQuery的unbind()
4. publish(type):执行订阅。类似jQuery的trigger();
当然上面的功能Cowboy大神只用了只行代码就实现了(基于jQuery):
(function($) {
//得到一个jQuery对象,以便使用其方法
var o = $({});
//为jQuery对象增加静态的方法
$.subscribe = function() {
o.bind.apply(o, arguments);
};
$.unsubscribe = function() {
o.unbind.apply(o, arguments);
};
$.publish = function() {
o.trigger.apply(o, arguments);
}
})(jQuery);
PubSub的一种实现的更多相关文章
- GCP消息队列Pubsub详解,简单好用还不用自己运维
我最新最全的文章都在南瓜慢说 www.pkslow.com,欢迎大家来喝茶! 1 简介 GCP的Pubsub是一种异步消息传递服务,可将生产事件的服务与处理事件的服务隔离开.消息队列的作用就不多作介绍 ...
- Redis info参数总结(转)
Redis官网对 info 已经讲解的比较清楚的,参考文档 . 可以看到,info的输出结果是分几块的,有Servers.Clients.Memory等等,通过info后面接这些参数,可以指定输出某一 ...
- 关于Redis info的参数总结
Redis官网对 info 已经讲解的比较清楚的,参考文档 .可以看到,info的输出结果是分几块的,有Servers.Clients.Memory等等,通过info后面接这些参数,可以指定输出某一块 ...
- (转)基于即时通信和LBS技术的位置感知服务(一):提出问题及解决方案
一.前言.提出问题 公司最近举行2011年度创新设计大赛,快年底了正打算写写2010年以来Android开发的心得与经验,正好同事出了个点子:假如A和B两个人分别在不同的地点,能不能实现这样的功能,让 ...
- redis info命令结果释疑
redis的性能数据这块用 info 命令就可以获取的比较全面了,下面是对info信息返回值的解释: # 参考:http://redis.io/commands/info # # # Server r ...
- vue的组件详解
什么是组件 组件(Component)是 Vue.js 最强大的功能之一.(好比电脑中的每一个元件(键盘,鼠标,CPU),它是一个具有独立的逻辑和功能或界面,同时又能根据规定的接口规则进行互相融合,变 ...
- Redis info参数总结
可以看到,info的输出结果是分几块的,有Servers.Clients.Memory等等,通过info后面接这些参数,可以指定输出某一块数据. 下面是针对info的输出在旁边注释了,因为对Redis ...
- JMS学习之理论基础
本文代码使用ActiveMq5.6 一.什么是JMS JMS(Java Message Service,Java消息服务)是一组Java应用程序接口(Java API),它提供创建.发送.接收.读取消 ...
- 组件通信之全局事件总线 & 消息订阅发布
全局事件总线 介绍 一种组件间通信的方式,适用于任意组件间通信. 在使用全局事件总线之前需要一些知识准备 所有组件实例的原型对象的原型对象就是 Vue 的原型对象,即VueComponent.prot ...
随机推荐
- 重温《js权威指南》 第7,8章
第七章 数组 数组是值的有序集合.js数组是无类型的,数组元素可以是任意类型,同一个数组中不同元素也可能有不同的类型.数组可以动态增长或缩减,创建时无须生命那个一个固定的大小并且数组大 ...
- IDL---ENVI
ENVI;启动envi file=envi_pickfile();选择文件dialog,返回值就为file ENVI_OPEN_FIle,file,r_fid=fid;根据文件名打开file,并且返回 ...
- DataGridView过滤区分大小写问题
DataTable上的过滤方法: 一.可以用DataTable.Select("条件"),返回DataRow[]格式的结果集. DataRow[] drArr = dt.Selec ...
- ES6入门之set和map
Set ES6提供了新的数据结构Set.它类似于数组,但是成员的值都是唯一的,没有重复的值. Set函数可以接受一个数组(或类似数组的对象)作为参数,用来初始化. // 例一 var set = ne ...
- log log4net用代码记录日志
log4net 用代码记录日志 今天在开发项目的时候,遇到跨域调用log4net中的类,出现了一个bug,提示LogImpl未标记可序列化,此时,我靠,麻烦了,这个类又不是咱们自己的,改源码我想应该 ...
- 第一个Linux驱动-流水灯【转】
转自:http://www.xuebuyuan.com/1856562.html 水平有限,描述不当之处请指出,转载请注明出处http://blog.csdn.net/vanbreaker/artic ...
- 19.allegro过孔设置[原创]
一.根据线宽设置过孔 在规则管理器下 --- --- ---- --- --- 二.设置原点 法1: -- -- 法二: 然后鼠标点选 ---option栏目在哪? --- 三:过孔问题1 当一个过孔 ...
- 【POJ】3398 Perfect Service
1. 题目描述某树形网络由$n, n \in [1, 10^4]$台计算机组成.现从中选择一些计算机作为服务器,使得每当普通计算机恰好与一台服务器连接(并且不超过一台).求需要指定服务器的最少数量 2 ...
- How to: Write Object Data to an XML File
This example writes the object from a class to an XML file using the XmlSerializer class. Namespace: ...
- lumen 使用 redis缓存
建议修改composer.json require 节点如下: "require": { "php": ">=5.5.9", &quo ...