`
yangsp1
  • 浏览: 49003 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

javascript事件处理机制——容易被我们遗忘的细节问题

    博客分类:
  • js
阅读更多
转自:http://www.iteye.com/topic/299320

最近在细读mootools core源代码中,在阅读到包装Event的时候,才发现,以前没对js事件处理处理机制深入的研究.遂,再次翻开尘封已久的java犀牛书,仔细的阅读了一把.

      我们在书写js程序的时候,在很大程度上,都是写客户端交互程序,所以大部分都可以事件有关.但是遗憾的是,这些事件处理程序的细节非常复杂.并且由于浏览器的不同,目前有四种不同的事件处理机制:

原始事件模型:这是最简单的一种事件处理模型,尽管功能有限,但所有的浏览器都支持它.可以把它看作0级DOM

标准事件模型:这是一种更强大更完整的事件模型.2级DOM标准对它进行了标准化.Netscape 6和Mozilla支持它

Internet Explorer事件模型:具有标准事件模型的许多特性。虽然Microsoft参与了2级DOM事件模型的创建。但IE7一下的版本都没实现2级模型的标准,坚持使用自己的专有的事件模型,IE8以上的不太清楚。这意味着,在ie下运行,要想使用高级事件处理特性的js程序必须编写特定的代码。

Netscape事件模型:这种模型现在已被标准模型取代.不过Netscape6为了向后兼容,还是继续实现了它.

接下来我们讨论这几种事件模型各自的特点以及如何实现一个通用健壮的事件处理程序在你的应用程序中

其实,我们在处理js事件问题的时候,我们就会很自然的去认为事件发生在目标元素上,那么它的事件处理代码就只会在你编写的特定处理函数上执行,例如:有这样一段代码:

   1. <div id="parent" style="height: 100px; width: 200px; background-color: yellow" onclick="parent(event)">  
   2.     <div id="child" style="height: 50px; width: 100px; background-color: gray" onclick="child(event)">  
   3.     </div>  
   4. </div>  

		<div id="parent" style="height: 100px; width: 200px; background-color: yellow" onclick="parent(event)">
			<div id="child" style="height: 50px; width: 100px; background-color: gray" onclick="child(event)">
			</div>
		</div>





   1. var parent = function(e) {  
   2.     alert('parent');  
   3. };  
   4. var child = function(e) {  
   5.     alert('child');  
   6. };  

			var parent = function(e) {
				alert('parent');
			};
			var child = function(e) {
				alert('child');
			};


在通常情况下, 我们会认为当点击parent的时候会打印出'parent'来,点击child的时候会打印出'child'来,但其实则不然,当你点击child的时候,会打印出'parent'和'child'.为什么结果和预想的不一样呢,那是因为我们对js的事件处理机制还建立在原始事件模型上.而浏览器却按照标准事件模型来处理的.那或许有人又有疑问,不是说IE不支持标准事件模型吗?为什么在IE下也会打印出'child'和parent.我们刚刚说了IE 用了自己专业的事件模型处理机制.但是却具有大部分标准事件模型的特性.所以会有这样的结果.那么我们来看看标准事件模型是怎样定义的吧!

     在原始事件模型中,浏览器把事件分派给发生事件的文档元素。如果该那个对象具有适合的事件处理程序,就运行这个程序。初次之外,不用执行其他的操作。但是,在标准事件模型中,情况则要复杂很多。当事件发生在目标元素时,目标的事件处理程序会被触发。此外,目标的每个祖先节点也有机会处理那个事件。事件传播分三个阶段完成。一:在捕捉(capturing)阶段,事件从Document对象沿着文档向下传播给目标节点。在这期间,目标对象的任何祖先节点都可以捕捉的事件并处理它。二:目标节点本身,这和原始时间模型相似,最后一个阶段是起泡阶段:在这个阶段,事件将从目标元素向上传播回 Document对象的文档层次。

     现在就不难理解为什么我们点击child的时候parent的点击事件处理函数也会被执行.这就是我们今天讨论的主题,被忽略的细节.虽然很多人都知道 javascript标准事件模型,但是,大部分人在写事件处理程序的时候却忽略了这些问题.所以.有些时候,在某段页面js代码中插入几段事件代码,却产生莫名的错误或者是与想要的结果有大大的出入.如果说你还没有遇到这样的问题,你应该为此而感到庆幸,那是因为你还没遇到过在一个父元素和子元素处理相同的事件.但这不代表你以后不会遇到.所以为了你代码的健壮性和易维护性.应该重构你的代码.下面这段代码是我的一个实现

   1. var Event = function(event) {  
   2.         this.event = event || window.event;  
   3.         this.target = (function() {  
   4.             var target = this.event.target || this.event.srcElement;  
   5.             while (target && target.nodeType == 3) {  
   6.                 target = target.parentNode;  
   7.             }     
   8.             return target;  
   9.         })();  
  10.           
  11.         this.stop = function(){  
  12.             return this.stopPropagation().preventDefault();  
  13.         }  
  14.           
  15.         this.stopPropagation = function(){  
  16.             if (this.event.stopPropagation) this.event.stopPropagation();  
  17.             else this.event.cancelBubble = true;  
  18.             return this;  
  19.         }  
  20.       
  21.         this.preventDefault = function(){  
  22.             if (this.event.preventDefault) this.event.preventDefault();  
  23.             else this.event.returnValue = false;  
  24.             return this;  
  25.         }  
  26. }  

var Event = function(event) {
		this.event = event || window.event;
		this.target = (function() {
			var target = this.event.target || this.event.srcElement;
			while (target && target.nodeType == 3) {
				target = target.parentNode;
			}	
			return target;
		})();
		
		this.stop = function(){
			return this.stopPropagation().preventDefault();
		}
		
		this.stopPropagation = function(){
			if (this.event.stopPropagation) this.event.stopPropagation();
			else this.event.cancelBubble = true;
			return this;
		}
	
		this.preventDefault = function(){
			if (this.event.preventDefault) this.event.preventDefault();
			else this.event.returnValue = false;
			return this;
		}
}



重构后的事件处理代码:

   1. var parent = function(e) {  
   2.     var event = new Event(e);  
   3.     alert('parent');  
   4. };  
   5. var child = function(e) {  
   6.     var event = new Event(e);  
   7.     alert('child');  
   8.     event.stopPropagation();  
   9. };  

			var parent = function(e) {
				var event = new Event(e);
				alert('parent');
			};
			var child = function(e) {
				var event = new Event(e);
				alert('child');
				event.stopPropagation();
			};


经过这样处理以后,便可保证在任何情况任何环境下照常运行
分享到:
评论

相关推荐

    利用JavaScript实现图片标注——SearchMapIdentityTask

    DOM(Document Object Model)是HTML和XML文档的结构化表示,通过JavaScript我们可以对DOM进行操作,比如添加、删除或修改元素。 在实现`SearchMapIdentityTask`时,我们可能需要以下关键知识点: 1. **事件监听**...

    JavaScript凌厉开发——Ext详解与实践 源码 源代码 part3

    JavaScript凌厉开发——Ext详解与实践 源码 源代码 part3 因为源代码比较大,压缩后76M左右 所以分为四个包上传

    JavaScript代码生成器——Coffee Script.pdf

    JavaScript代码生成器——Coffee Script CoffeeScript是一种基于Ruby语言的编程语言,旨在通过简洁的编码方式生成JavaScript代码。它结合了Ruby的简洁和JavaScript的灵活性,使开发者可以通过简洁易读的语法撰写...

    网页模板——扣代码工具 javascript事件 捕获者2.0.zip

    JavaScript事件捕获机制是理解JavaScript事件处理模型的关键部分,它允许我们对页面上的元素进行响应,例如点击、鼠标移动或者键盘输入等。下面将详细阐述这两个主题。 首先,网页模板是一种预先设计的网页结构,...

    《javascript》——event对象与事件

    《JavaScript》——Event 对象与事件 Event 对象是 JavaScript 中的一个基本概念,它代表了用户与 Web 页面的交互行为,例如点击、鼠标移动、键盘输入等。Event 对象的属性和方法可以帮助开发者更好地处理用户的...

    JavaScript特效代码集——rar

    2. **鼠标事件类**:JavaScript中的事件处理是用户交互的核心。代码集中包含了鼠标点击(click)、移动(mousemove)、悬停(mouseover/mouseout)等各种事件的处理方法,可以帮助你创建响应式和互动性强的网页元素...

    JavaScript网页开发——体验式学习教程.pdf

    事件监听器、事件处理函数和事件传播机制是这部分的重点。 AJAX(异步JavaScript和XML)技术使得无需刷新页面就能与服务器交换数据并更新部分网页内容。这在现代Web应用中非常常见,提高用户体验。教程将涵盖...

    javascript小工具之——cookie操作

    在这些任务中,Cookie是JavaScript用来存储和检索少量信息的一种机制,特别是在用户会话管理、个性化设置和跟踪用户行为时非常有用。这篇博客“javascript小工具之——cookie操作”很可能会探讨如何利用JavaScript...

    Javascript中的神器——Promise

    回调函数真正的问题在于他剥夺了我们使用 return 和 throw 这些关键字的能力。而 Promise 很好地解决了这一切。 2015 年 6 月,ECMAScript 6 的正式版 终于发布了。 ECMAScript 是 JavaScript 语言的国际标准,...

    教你一天玩转JavaScript(三)——使用JavaScript完成图片轮播的效果

    通过研究和实践这个示例,你将对JavaScript的DOM操作、事件处理、计时器以及与CSS结合实现动态效果有更深入的理解。动手实践是学习JavaScript最好的方式,祝你在探索JavaScript的道路上不断进步!

    教你一天玩转JavaScript(二)——完成对注册页面的数据的简单校验

    JavaScript的事件监听机制可以帮助我们实现这一功能。例如,可以监听`onblur`事件,当用户离开输入框时进行校验: ```javascript document.getElementById('emailInput').addEventListener('blur', function() { ...

    教你一天玩转JavaScript(八)——使用JavaScript完成省市联动的效果

    接着,我们需要掌握JavaScript事件处理。在本例中,关键事件是`change`事件,当用户在省份下拉框中做出选择时触发。我们将在省份下拉框上添加事件监听器,以便在选中新的省份时更新城市列表。 然后,数据的存储也是...

    js处理文件——文件读写例子.zip

    它有多个事件处理方法,如`onloadstart`, `onprogress`, `onload`, `onerror`, 和 `onloadend`,用于监听文件读取的各个阶段。例如,当文件读取成功时,`onload`事件会被触发,可以在这个事件的回调函数中处理读取到...

    JavaScript例子——计算

    6. **事件处理**:在网页环境中,JavaScript常与用户交互,通过事件监听(addEventListener)实现点击按钮触发计算等功能。例如: ```html &lt;button id="calcBtn"&gt;Calculate document.getElementById('calcBtn')....

    教你一天玩转JavaScript(七)——使用JavaScript完成复选框的全选和全不选的效果

    5. **事件处理函数**:在事件处理函数中,我们可以根据需求实现全选或全不选的操作。例如,如果全选按钮被点击,遍历所有复选框并将其`checked`属性设为`true`。 6. **反向操作**:对于全不选功能,只需要将全选...

    Applet与Javascript的对话——让你的Javascript代码和Java Applet融洽地合作.pdf

    标题中的“Applet与Javascript的对话——让你的Javascript代码和Java Applet融洽地合作”指的是一种技术实现,即如何让JavaScript与Java Applet在Web应用中进行交互。这两种技术在早期Web开发中常常结合使用,以利用...

    JavaScript动态网页开发详解——源文件

    在"JavaScript动态网页开发详解——源文件"中,我们可以深入学习到JavaScript在网页开发中的应用技巧。此资料可能包含了JQUERY的官方实例全集,jQuery是一个高效、简洁且富有创造性的JavaScript库,它极大地简化了...

    XMPP高级编程——使用JavaScript和jQuery

    4. **事件处理**:jQuery的事件处理机制使得监听和响应XMPP事件变得轻松,比如在线状态变化、消息到达等。 **XMPP应用开发**: 1. **Strophe.js**:一个轻量级的JavaScript库,专门用于XMPP通信,它简化了与XMPP...

    精通JavaScript ——动态网页(实例版)详尽实例

    此外,JavaScript的事件处理也是重要内容。从简单的按钮点击到复杂的拖放操作,JavaScript事件处理使得网页能响应用户的各种交互。书中将展示如何绑定事件监听器、处理事件对象,以及如何避免事件冒泡和捕获的问题。...

    asp.net中TextBox获得焦点和失去焦点——客户端JavaScript事件

    首先,我们要理解JavaScript事件。JavaScript是一种运行在客户端浏览器上的脚本语言,它能监听用户行为和页面状态的变化,然后触发相应的函数。在TextBox中,两个关键的事件是`focus`和`blur`。 1. `focus`事件:当...

Global site tag (gtag.js) - Google Analytics