JavaScript中事件处理
先看看下面一道题目,请评价以下代码并给出改进意见:
if (window.addEventListener) {//标准浏览器
var addListener = function(el, type, listener, useCapture) {
el.addEventListener(type, listener, useCapture);
};
} else if (document.all) {//IE
addListener = function(el, type, listener) {
el.attachEvent("on" + type, function() {
listener.apply(el);
});
}
}
1)不应该在if和else语句中声明addListener函数,应该先声明;
2)不需要使用window.addEventListener或document.all来进行检测浏览器,应该使用能力检测;
3)attachEvent在IE中有this指向问题,会指向window,虽然上面的代码做了指向处理,但是匿名函数不能做detachEvent解绑
改进后的代码稍后加上。
一、冒泡与捕获
使用过addEventListener方法的会发现最后一个参数“useCapture”,用于控制是捕获还是冒泡。
何为冒泡、捕获,请看下面的例子:查看在线代码,在线代码中用了三层。
<div id="click1">
<div id="click2">事件</div>
</div>
1)Netscape主张元素1的事件首先发生,这种事件发生顺序被称为捕获型,如下图所示:
| |
---------------| |-----------------
| click1 | | |
| -----------| |----------- |
| |click2 \ / | |
| ------------------------- |
| Event CAPTURING |
-----------------------------------
2)微软则保持元素2具有优先权,这种事件顺序被称为冒泡型,如下图所示:
/ \
---------------| |-----------------
| click1 | | |
| -----------| |----------- |
| |click2 | | | |
| ------------------------- |
| Event BUBBLING |
-----------------------------------
3)W3C选择了一个择中的方案。任何发生在w3c事件模型中的事件,首是进入捕获阶段,直到达到目标元素,再进入冒泡阶段,如下图所示:
| | / \
---------------| |--| |------------
| click1 | | | | |
| -----------| |--| |------ |
| |click2 \ / | | | |
| ------------------------- |
| W3C event model |
-----------------------------------
4)阻止冒泡,很多时候是不想触发父级的相同事件的,那么就需要阻止这种行为。
W3C方:event.stopPropagation()。(chrome、firefox、safrai等)
IE方:event.cancelBubble设置为true。
经过我的在线测试,stopPropagation这个方法不能阻止捕获。
这里顺带说下阻止默认事件的方法,何为默认事件?就比如a标签设置了href,就会做跳转,这里阻止它跳转。
W3C方:event.preventDefault(),但只有event的cancelable属性为true时才能使用。
IE方:event.returnValue设置为false。
二、事件系统
浏览器提供了3种层次的API。
1)最原始的是写在元素标签内
2)以el.onXXX=function绑定的方式,通称为DOM0事件系统。
3)一个元素的同一类型事件可以绑定多个回调,通称为DOM2事件系统。
IE与W3C依旧不同,语法如下:
| 序号 | 操作与对象 | IE方 | W3C方 |
|
1 |
绑定事件 |
el.attachEvent("on"+type, callback) | el.addEventListener(type,callback,[useCapture]) |
|
2 |
卸载事件 |
el.detachEvent("on"+type,callback) | el.removeEventListener(type,callback,[useCapture]) |
| 3 | 创建事件 | document.createEventObject() |
document.createEvent(types) 创建事件(过时) event.initEvent() 初始化事件(过时) new Event(types) |
|
4 |
派发事件 |
el.fireEvent(type,event) | el.dispatchEvent(event) |
| 5 | event属性 | srcElement:等于target,默认目标 |
currentTarget:其事件处理程序当前正处理事件的元素 target:事件的目标 |
| 6 | event方法 |
returnValue:等于preventDefault() cancelBubble:设为true等于stopPropagation() |
preventDefault():阻止默认行为 stopPropagation():阻止冒泡 |
|
7 |
type |
被触发的事件类型,需要“on”前缀 | 被触发的事件类型 |
|
8 |
事件执行顺序 |
与添加顺序相反 | 与添加顺序一致 |
| 9 |
匿名函数 |
无法移除 | 无法移除 |
|
10 |
this |
window | 当前绑定的元素 |
注意第8点,在测试代码中,我绑定了两个相同的“click”事件, 查看在线完整代码。
var func1 = function(e) {
alert(1); //测试执行顺序
};
var func2 = function(e) {
alert(2);
};
var type = 'click';
bind(ele, type, func1);
bind(ele, type, func2);
在IE中先弹出2,再弹出1。而在chrome中先弹出1,再弹出2。
注意上面的第9点和第10点,上面那道题目中要解决的就是这个问题。下面的代码是个片段,查看在线完整代码。
var bind = function(ele, type, callback) {
if (!ele[type + "event"]) {
ele[type + "event"] = {}; //声明一个空对象,缓存事件
}
var name = callback.toString();
if (!ele[type + "event"][name]) {
var handler = function(event) {
//可以做更多event封装操作
var ev = event || window.event;
callback.call(ele, ev);
};
ele[type + "event"][name] = handler; //做个临时变量
}
if (ele.addEventListener) {
ele.addEventListener(type, handler, false);
} else if (ele.attachEvent) {
ele.attachEvent('on' + type, handler);
}
}
var unbind = function(ele, type, callback) {
var handler = ele[type + "event"][callback.toString()]; //读取临时变量
if (ele.removeEventListener) {
ele.removeEventListener(type, handler);
} else if (ele.detachEvent) {
ele.detachEvent('on' + type, handler);
}
}
上面的代码还比较粗糙,仅仅是用于演示一下。方法有很多,自己可以揣摩。
我看到网上有人直接不用attachEvent,将相应的事件保存在一个数组中,当detachEvent的时候做splice数组的操作。
大家也可以参考下一些成熟的类库,例如jQuery1.8.3版本,2642行的event.add封装,2757行的event.remove封装,3485行的on封装。
也可以参考Dean Wdwards写的addEvent方法,这是Prototype时代早期出现的一个事件系统,jQuery事件系统的源头。
参考资料:
JavaScript中事件处理的更多相关文章
- [转]理解JavaScript中的事件处理
什么是事件? 事件(Event)是JavaScript应用跳动的心脏 ,也是把所有东西粘在一起的胶水.当我们与浏览器中 Web 页面进行某些类型的交互时,事件就发生了.事件可能是用户在某些内容上的点击 ...
- 理解JavaScript中的事件处理
什么是事件? 事件(Event)是JavaScript应用跳动的心脏 ,也是把所有东西粘在一起的胶水.当我们与浏览器中 Web 页面进行某些类型的交互时,事件就发生了.事件可能是用户在某些内容上的点击 ...
- 理解JavaScript中的事件处理 阻止冒泡event.stopPropagation();
原文地址:http://www.cnblogs.com/binyong/articles/1750263.html 这篇文章对于了解Javascript的事件处理机制非常好,将它全文转载于此,以备不时 ...
- 详解JavaScript中的事件处理
在漫长的演变史,我们已经告别了内嵌式的事件处理方式(直接将事件处理器放在HTML元素之内来使用),今天的事件,它已是DOM的重要组成部分,遗憾的是,IE继续保留它最早在IE4.0中实现的事件模型,以后 ...
- 关于javascript中的this关键字
this是非常强大的一个关键字,但是如果你不了解它,可能很难正确的使用它. 下面我解释一下如果在事件处理中使用this. 首先我们讨论一下下面这个函数中的this关联到什么. function doS ...
- 理解JavaScript中的作用域和上下文
JavaScript对于作用域(Scope)和上下文(Context)的实现是这门语言的一个非常独到的地方,部分归功于其独特的灵活性. 函数可以接收不同的的上下文和作用域.这些概念为JavaScrip ...
- javascript 中的事件机制
1.javascript中的事件. 事件流 javascript中的事件是以一种流的形式存在的. 一个事件会也有多个元素同时响应. 有时候这不是我们想要的效果, 我们只是需要某个特定的元素相应我们的绑 ...
- 20个优秀的 JavaScript 键盘事件处理库
键盘事件是 Web 开发中最常用的事件之一,通过对键盘事件的捕获和处理可以提高网站的易用性和交互体验.下面,我们向大家介绍收集的20款优秀的 JavaScript 键盘事件处理库,帮助开发人员轻松处理 ...
- JavaScript中的this关键字
在JavaScript中,函数的this关键字的行为与其他语言相比有很多不同.在JavaScript的严格模式和非严格模式下也略有区别. 在绝大多数情况下,函数的调用方式决定了this的值.this不 ...
随机推荐
- 03.SQLServer性能优化之---存储优化系列
汇总篇:http://www.cnblogs.com/dunitian/p/4822808.html#tsql 概 述:http://www.cnblogs.com/dunitian/p/60413 ...
- opencv中Mat与IplImage,CVMat类型之间转换
opencv中对图像的处理是最基本的操作,一般的图像类型为IplImage类型,但是当我们对图像进行处理的时候,多数都是对像素矩阵进行处理,所以这三个类型之间的转换会对我们的工作带来便利. Mat类型 ...
- ASP.NET Core 中文文档 第四章 MVC(3.8)视图中的依赖注入
原文:Dependency injection into views 作者:Steve Smith 翻译:姚阿勇(Dr.Yao) 校对:孟帅洋(书缘) ASP.NET Core 支持在视图中使用 依赖 ...
- C#异步编程(二)
async和await结构 序 前篇博客异步编程系列(一) 已经介绍了何谓异步编程,这篇主要介绍怎么实现异步编程,主要通过C#5.0引入的async/await来实现. BeginInvoke和End ...
- ASP.NET WebApi OWIN 实现 OAuth 2.0
OAuth(开放授权)是一个开放标准,允许用户让第三方应用访问该用户在某一网站上存储的私密的资源(如照片,视频,联系人列表),而无需将用户名和密码提供给第三方应用. OAuth 允许用户提供一个令牌, ...
- 通过AngularJS实现前端与后台的数据对接(一)——预备工作篇
最近,笔者在做一个项目:使用AngularJS,从而实现前端与后台的数据对接.笔者这是第一次做前端与后台的数据对接的工作,因此遇到了许多问题.笔者在这些问题中,总结了一些如何实现前端与后台的数据对接的 ...
- C#反序列化XML异常:在 XML文档(0, 0)中有一个错误“缺少根元素”
Q: 在反序列化 Xml 字符串为 Xml 对象时,抛出如下异常. 即在 XML文档(0, 0)中有一个错误:缺少根元素. A: 首先看下代码: StringBuilder sb = new Stri ...
- bzoj3208--记忆化搜索
题目大意: 花花山峰峦起伏,峰顶常年被雪,Memphis打算帮花花山风景区的人员开发一个滑雪项目. 我们可以把风景区看作一个n*n的地图,每个点有它的初始高度,滑雪只能从高处往低处滑[严格大于] ...
- python 数据类型 ---文件一
1.文件的操作流程: 打开(open), 操作(read,write), 关闭(close) 下面分别用三种方式打开文件,r,w,a 模式 . "a"模式将不会覆盖原来的文件内容, ...
- css知多少之绝对定位小记
一.position定位常见属性 对于属性position来说,属性值有static/relative/absolute/fixed/inherit以下只对绝对定位position:absolute详 ...