JavaScript事件流(冒泡事件与捕获事件)
来源:互联网 发布:windows部署添加驱动 编辑:程序博客网 时间:2024/05/19 20:00
事件流
事件流是在浏览器发展到第四代(IE4及Netscape Communicator4)时出现的概念,事件流描述的是从页面接受事件的顺序。但有意思的是,IE和Netscape开发团队居然提出了差不多是完全相反的事件流的概念。IE的事件流时事件冒泡流,而Netscape的事件流时事件捕获流。
事件冒泡
事件冒泡(event bubbling),即事件开始时有最具体的元素(文档中嵌套层次最深的那个节点,也可以说是真实触发节点)接受,然后逐级向上传播到较为不具体的节点(文档:document)
<!DOCUTYPE html><html> <head> </head> <body> <div id="div">Click Me</div> </body><html>
点击div整个事件的传播
IE9、Firefox、Chrome、Safari会一直冒泡到window对象
事件捕获
事件捕获(event capturing),即不太具体的节点应该更早接收到事件,而最具体的节点(真实触发节点)应该最后接收到事件,仍以之前的HTML为例,事件传递顺序:
DOM事件流
“DOM2级事件”规定的事件流包括三个阶段:事件捕获阶段、处于目标阶段和事件冒泡阶段。首先发生的是事件捕获,为截获事件提供了机会。然后是实际的目标接受到事件。最后一个阶段是冒泡阶段,可以在这个阶段对事件作出相应。 仍以前面的HTML页面为例,事件触发顺序:
在DOM2事件流中,实际的目标(div元素)在捕获阶段不会接受到事件。意味着在捕获阶段,事件从document到
<html>
再到<body>
后就停止了。下一个阶段是“处于目标”阶段,于是事件在<div>
上发生,并在事件处理中被看成冒泡阶段的一部分。然后,冒泡阶段发生,事件又传播回文档。多数支持DOM事件流的浏览器都实现了一种特定的行为;即使“DOM2级事件”规范明确要求捕获阶段不会涉及事件目标,但IE9、Safari、Chrome、Firefox和Opera 9.5及更高版本都会在捕获阶段触发事件对象上的事件。结果,就是有两个机会在目标对象上面操作事件。
DOM2级事件处理程序
“DOM 2级事件”定义了两个方法,addEventListener()绑定指定事件处理程序、removeEventListener()删除指定事件处理程序。所有DOM节点中都包含这两个方法,都接受3个参数:(event,function,boolean)。最后布尔值参数如果是true,表示在捕获阶段调用事件处理函数;如果为false,表示在冒泡阶段调用事件处理程序;默认为false。
var btn = document.getElementById("myBtn"); btn.addEventListener("click", function(){ alert(this.id); }, false);
上面的代码为一个按钮添加了onclick事件处理程序,而且该事件会在冒泡阶段被触发(因为最后一个参数是false)。使用DOM2级方法添加事件处理程序的主要好处是可以添加多个事件处理程序。来看下面的例子。
var btn = document.getElementById("myBtn"); btn.addEventListener("click", function(){ alert(this.id); }, false); btn.addEventListener("click", function(){ alert("Hello world!"); }, false);
这里为按钮添加了两个事件处理程序。这两个事件处理程序会按照添加它们的顺序触发,因此首先会显示元素的ID,其次会显示”Hello world!”消息。
通过addEventListener()添加的事件处理程序只能使用removeEventListener()来移除;移除时传入的参数与添加处理程序时使用的参数相同。这也意味着通过addEventListener()添加的匿名函数将无法移除,如下面的例子所示。
var btn = document.getElementById("myBtn"); btn.addEventListener("click", function(){ alert(this.id);}, false); //这里省略了其他代码 btn.removeEventListener("click", function(){ //没有用! alert(this.id); }, false);
在这个例子中,我们使用addEventListener()添加了一个事件处理程序。虽然调用removeEventListener()时看似使用了相同的参数,但实际上,第二个参数与传入addEventListener()中的那一个是完全不同的函数。而传入removeEventListener()中的事件处理程序函数必须与传入addEventListener()中的相同,如下面的例子所示。
var btn = document.getElementById("myBtn"); var handler = function(){ alert(this.id); }; btn.addEventListener("click", handler, false); //这里省略了其他代码 btn.removeEventListener("click", handler, false); //有效!
重写后的这个例子没有问题,是因为在addEventListener()和removeEventListener()中使用了相同的函数。
多数情况下,都是将事件处理程序添加到事件流的冒泡阶段,这样可以最大限度地兼容各种浏览器。最好只在需要在事件到达目标之前截获它的时候将事件处理程序添加到捕获阶段。如果不是特别需要,我们不建议在事件捕获阶段注册事件处理程序。
事件对象
兼容DOM的浏览器会将一个event对象传入到事件处理程序中。无论指定事件处理程序时使用什么方法(DOM0级或DOM2级),都会传入event对象。
var btn = document.getElementById("myBtn"); btn.onclick = function(event){ alert(event.type); //"click" }; btn.addEventListener("click", function(event){ alert(event.type); //"click" }, false);
- 事件对象的属性
在事件处理程序内部,对象this始终等于currentTarget的值,而target则只包含事件的实际目标。如果直接将事件处理程序指定给了目标元素,则this、currentTarget和target包含相同的值。来看下面的例子。
var btn = document.getElementById("myBtn"); btn.onclick = function(event){ alert(event.currentTarget === this); //true alert(event.target === this); //true };
部分内容出自《JavaScript高级程序设计》第三版,欢迎指出错误点
- JavaScript事件流(冒泡事件与捕获事件)
- javascript事件冒泡与捕获
- JavaScript事件捕获与事件冒泡原理
- javascript事件冒泡与事件捕获
- javascript事件冒泡与事件捕获
- 浅谈JavaScript事件冒泡与事件捕获
- JavaScript 事件冒泡 捕获
- 事件冒泡与事件捕获
- 事件捕获与事件冒泡
- 事件冒泡与事件捕获
- 事件捕获与事件冒泡
- 事件冒泡与事件捕获
- 事件捕获与事件冒泡
- 事件捕获与事件冒泡
- 事件冒泡与事件捕获
- 事件冒泡与事件捕获
- 事件冒泡与事件捕获
- 事件冒泡与事件捕获
- vi/vim 使用解说
- 全屏到任意宽高的Dialog的写法
- 如何停止焦虑
- Android开源项目
- 修改wamp中默认的数据库密码
- JavaScript事件流(冒泡事件与捕获事件)
- Chapter 10 Hidden Markov Model
- 2017秋高级软件工程学习总结-冯涛
- 如何在GITHUB下载单个文件夹
- Matlab矩阵行列变换
- python学习—Day21—json
- 如何下载Google官方Git的Android任何版本源码Android 7.1,Android 8.0,8.1…
- VNC的安装部署
- NEFU OJ 1247 平静的小镇