设为首页 收藏本站
查看: 878|回复: 0

[经验分享] 自己实现一个javascript事件模块

[复制链接]

尚未签到

发表于 2017-2-23 09:25:02 | 显示全部楼层 |阅读模式
nodejs中的事件模块
  nodejs中有一个events模块,用来给别的函数对象提供绑定事件、触发事件的能力。这个别的函数的对象,我把它叫做事件宿主对象(非权威叫法),其原理是把宿主函数的原型链指向时间模块的一个对象,做一个函数继承,让宿主函数也拥有处理事件的能力
  
使用nodejs事件模块的demo如下:
var EventEmitter = require('events');  
var util = require('util');
  

  
function MyEmitter() {
  EventEmitter.call(this);
  
}
  
util.inherits(MyEmitter, EventEmitter);
  

  
var myEmitter = new MyEmitter();
  
myEmitter.on("hehe",function(){
  console.log("hehe")
  
})
  
myEmitter.on("hehe",function(){
  console.log("haha")
  
})
  
myEmitter.emit("hehe");
  
console.log(myEmitter.listeners("hehe"));
  看node events的api http://nodejs.cn/api/events.html ,一个事件模块需要具备如下的基本功能:


  • 添加绑定事件
  • 添加绑定一次性事件
  • 移除事件监听
  • 触发事件
  • 其他需要的扩展
事件模块设计
  俗话说得好,不会造轮子的车手不是一个老司机!自己实现一个简单的事件系统,实现上面几个基本的功能就行。
  
捋一捋思路,我们需要先实现一个事件类,用来创建不同名称的事件对象,例如click事件对象,hehe事件对象等,并在对象内部维持一个事件被触发时的处理函数列表。
  
对于同一个事件宿主对象而言,对它绑定的事件应该有如下特点:


  • 同一种类型的事件对象只需要有一个
  • 处理函数需要有多个
  • 对于同一个事件,重复绑定同一个事件处理函数应该是无效的
  于是得到如下结构:
/**  
* 定义一个事件单元类
  
* @param eventname
  
* @constructor
  
*/
  
function EventMeta(eventname){
  this.name=eventname;
  this.handlerMap={};
  this.handlerList=[];
  
}
  每个事件宿主对象都应该自己持有一个map,用来保存自己的事件单元对象,加上上面所说的,绑定,一次性绑定、解绑、触发,宿主函数的原型应该被指向这样一个对象:
var prototype={  addEventListener:addEventListener,
  once:once,
  removeEventListener:removeEventListener,
  trigger:trigger,
  getEventHandlerMap:getEventHandlerMap
  }
  为了让宿主函数拥有这些属性,需要一个函数来为宿主函数指一下prototype属性,这里要注意的是,一定要调用这个函数之后,再扩展宿主函数自身的prototype属性,否则会被覆盖
function czEvent(fn){  if(typeof fn!=="function") return;
  fn.prototype=Object.create(prototype);
  }
  用户使用的时候这样调用一下就可以了
var czEvent=require("../czEvent");  
function MyEmitter() {
  
}
  
czEvent(MyEmitter);

依次实现事件函数

addEventListener
    /**  
     * 绑定事件监听
  
     * @param type
  
     * @param handler
  
     */
  function addEventListener(eventname,handler){
  var EventHandlerMap=this.getEventHandlerMap();
  var meta=EventHandlerMap[eventname];
  if(!meta){
  meta=EventHandlerMap[eventname]=new EventMeta(eventname);
  }
  var handlerId=handler.__handlerId;
  if(!handlerId){
  EventHandlerMap.__handlerId++;
  handler.__handlerId=EventHandlerMap.__handlerId;
  handlerId=handler.__handlerId;
  }
  //对于同一事件的同一处理函数,不重复绑定
  if(!meta.handlerMap[handlerId]){
  meta.handlerMap[handlerId]=handler;
  meta.handlerList.push(handler);
  }
  }

once
    /**  
     * 绑定只执行一次的handler
  
     * @param type
  
     * @param handler
  
     */
  function once(eventname,handler){
  handler.__once=true;
  this.addEventListener(eventname,handler);
  }

removeEventListener
    /**  
     * 移除事件监听
  
     * @param type
  
     * @param handdler
  
     */
  function removeEventListener(eventname,handdler){
  var EventHandlerMap=this.getEventHandlerMap();
  var meta=EventHandlerMap[eventname];
  if(!meta) return;
  //移除一个handler的绑定
  if(handdler && handdler.__handlerId){
  var index=meta.handlerList.indexOf(handdler);
  if(index>-1){
  meta.handlerList.splice(index,1);
  delete meta.handlerMap[handdler.__handlerId];
  return;
  }
  }
  //移除所有handler
  meta.handlerMap={};
  meta.handlerList.length=0;
  }

trigger
    /**  
     * 触发事件
  
     * @param eventname
  
     */
  function trigger(){
  var EventHandlerMap=this.getEventHandlerMap();
  var args=[];
  var that=this;
  for(var i=0;i<arguments.length;i++){
  args.push(arguments);
  }
  var eventname=args.splice(0,1);
  var meta=EventHandlerMap[eventname];
  if(!meta) return;
  var onceHandlerList=[];
  //依次同步执行handler
  meta.handlerList.forEach(function(handler){
  if(handler.__once){
  onceHandlerList.push(handler)
  }
  handler.apply(this,args);
  });
  //清除绑定为once的事件
  onceHandlerList.forEach(function(handler){
  that.removeEventListener(eventname,handler);
  });
  }

getEventHandlerMap
    function getEventHandlerMap(){  var EventHandlerMap=this.__EventHandlerMap;
  if(EventHandlerMap) return EventHandlerMap;
  return this.__EventHandlerMap={__handlerId:0};
  }

包装
  学(chao)习(xi)一下jquery的包装技术,让模块支持commonJs和amd
/**  
* Created by czzou on 2016/6/30.
  
*/
  
( function( global, factory ) {
  &quot;use strict&quot;;
  if ( typeof module === &quot;object&quot; && typeof module.exports === &quot;object&quot; ) {
  module.exports = factory( global, true );
  } else {
  factory( global );
  }
  
}(typeof window !== &quot;undefined&quot; ? window : this,function(window,noGlobal){
  //把上面的代码copy进来
  if ( typeof define === &quot;function&quot; && define.amd ) {
  define( &quot;czEvent&quot;, [], function() {
  return czEvent;
  } );
  }
  if ( !noGlobal ) {
  window.czEvent = czEvent;
  }
  return czEvent;
  
}))

扩展
  如有需求,还可以在此基础上进行一些扩展,比如nodejs的events模块支持的handler函数个数上限、handler函数抛出异常时触发的error事件等,另外,简单改写一下czEvent函数,还可以让事件模块支持给对象赋予事件处理能力的功能
    function czEvent(fn,asObj){  if(asObj){
  for(var key in prototype){
  Object.defineProperty(fn,key,{
  value:prototype[key]
  })
  }
  return;
  }
  if(typeof fn!==&quot;function&quot;) return;
  fn.prototype=Object.create(prototype);
  }

测试
  分别测试nodejs的events模块,czEvent的commonJs用法,amd用法以及直接引用用法,demo就不在这里一一列举了~
  
DSC0000.png
  github地址:https://github.com/zouchengzhuo/czevent
  本文转自我的个人站点:http://zoucz.com/blog/2016/07/01/czevent/
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-345985-1-1.html 上篇帖子: 个人网站架构设计(二) 下篇帖子: 使用 JavaScriptService 在.NET Core 里实现DES加密算法
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表