`

jQuery事件处理:return false与event.preventDefault();

 
阅读更多

可能在你刚开始学习关于jQuery事件处理时,看到的第一个例子就是关于如何阻止浏览器执行默认行为,比如下面这段演示click事件的代码:

  1. $("a.toggle").click(function () {  
  2.     $("#mydiv").toggle();  
  3.     return false; // Prevent browser from visiting `#`  
  4. });  

这个函数使用toggle来显示或者隐藏#mydiv,然后阻止浏览器继续访问href中指定的链接。

像上面这样的例子会让用户养成使用“return false”来阻止浏览器执行默认行为的坏习惯,在这篇文章里,我将会讨论关于阻止浏览器执行默认行为的两个非常重要的主题:

  • 选择正确的方法: return false还是preventDefault,stopPropagation或者stopImmediatePropagation
  • 选择合适的位置,开始,结束,还是中间某个地方:你应该在事件回调的哪个部分取消浏览器执行默认行为?

注意:当我在这篇文章中提到event bubbling(事件冒泡),我想表达的是大部分事件都是先在初始DOM上触发,然后再通过DOM树往上,在每一级父元素上触发,事件不会在兄弟节点或 是子节点上冒泡(当事件向下冒泡时,我们叫它事件捕捉(event capturing)),你可以在这里了解更多关于事件起泡和捕捉的介绍。

选择正确的方法

“return false”之所以被误用的如此厉害,是因为它看起来像是完成了我们交给它的工作,浏览器不会再将我们重定向到href中的链接,表单也不会被继续提交,但这么做到底有什么不对呢?

”return false“到底做了什么?

当你每次调用”return false“的时候,它实际上做了3件事情:

  • event.preventDefault();
  • event.stopPropagation();
  • 停止回调函数执行并立即返回。

“等等”,你叫了起来!我只是想让浏览器停止继续执行默认行为而已,我不需要它去做另外2件事。

这3件事中用来阻止浏览器继续执行默认行为的只有preventDefault,除非你想要停止事件冒泡,否则使用return false会为你的代码埋下很大的隐患,让我们通过一个真实的例子来看看这样的误用会造成什么后果:

这是我们用来演示的HTML:

  1. <div class="post">  
  2.   <h2><a href="http://heikezhi.com/path/to/page">My Page</a></h2>  
  3.   <div class="content">  
  4.     Teaser text...  
  5.   </div>  
  6. </div>  
  7. <div class="post">  
  8.   <h2><a href="http://heikezhi.com/path/to/other_page">My Other Page</a></h2>  
  9.   <div class="content">  
  10.     Teaser text...  
  11.   </div>  
  12. </div>  

现在假设我们想要在用户点击文章标题时,将文章动态载入到div.contentd中:

  1. jQuery(document).ready(function ($) {  
  2.   $("div.post h2 a").click(function () {  
  3.     var a    = $(this),  
  4.     href = a.attr('href'), // Let jQuery normalize `href`,  
  5.     content  = a.parent().next();  
  6.     content.load(href + " #content");  
  7.     return false; // "cancel" the default behavior of following the link  
  8.   });  
  9. });  

这段代码可以正常工作(至少目前是),但如果我们顺着这个思路继续,如果我想要在用户点击了一个div.post元素(或者任何一个它的子元素)时,给它加上一个active类,我就需要给div.post增加了一个click回调:

  1. // Inside Document Ready:  
  2. var posts = $("div.post");  
  3.   posts.click(function () {  
  4.   // Remove active from all div.post  
  5.   posts.removeClass("active");  
  6.   // Add it back to this one  
  7.   $(this).addClass("active");  
  8. });  

现在,如果我们点击一个帖子的标题,这段代码会工作吗?答案是不会,因为我们在标题的click回调里使用了return false而不是我们应该使用的,”return false“等于event.preventDefault();加event.stopPropagation();,所以事件冒泡就被终止 了,click事件不会被冒泡到div.post上,我们为它添加的事件回调当然也就不会被调用了。

如果我们把它和live或者delegate事件混在一起使用时,情况就更糟了。

  1. $("a").click(function () {  
  2.   // do something  
  3.   return false;  
  4. });  
  5.   
  6. $("a").live("click", function () {  
  7.   // THIS WON'T FIRE  
  8. });  

那么我们真正需要的是什么呢?

preventDefault()

大多数情况下,当你使用return false时,你其实真正需要的是e.preventDefault()。要使用e.preventDefault,你需要确保你传递了event参数到你的回掉函数中(在这个例子里,就是那个e):

  1. $("a").click(function (e) {  
  2.   // e == our event data   
  3.   e.preventDefault();  
  4. });  

它会替我们完成所有工作,但不会阻止父节点继续处理事件,要记住,你放在代码中的限制越少,你的代码就越灵活,也就越易于维护。

stopPropagation()

但有些情况下,你有可能需要停止事件冒泡,让我们看看下面的例子:

  1. <div class="post">  
  2.     Normal text and then a <a href="http://heikezhi.com/path">link</a> and then more text.  
  3. </div>  

现在,让我们假设如果你点了div上除了a链接之外的地方,我们希望能发生点什么事情(比如改变下背景什么的),但是不能影响用户点击a链接的行为(从可用性的角度,这个例子不怎么好,你可能不希望用户点击别的地方时发生任何事情)。

  1. $("div.post").click(function () {  
  2.   // Do the first thing;  
  3. });  
  4.   
  5. $("div.post a").click(function (e) {  
  6.   // Don't cancel the browser's default action  
  7.   // and don't bubble this event!  
  8.   e.stopPropagation();  
  9. });  

在这种情况下,如果我们使用return false,div的click事件不会被触发,但是用户也不会到达他们点的那个链接。

stopImmediatePropagation()

这个方法会停止一个事件继续执行,即使当前的对象上还绑定了其它处理函数,所有绑定在一个对象上的事件会按绑定顺序执行,看看下面的例子:

  1. $("div a").click(function () {  
  2.   // Do something  
  3. });  
  4.   
  5. $("div a").click(function (e) {  
  6.   // Do something else  
  7.   e.stopImmediatePropagation();  
  8. });  
  9.   
  10. $("div a").click(function () {  
  11.   // THIS NEVER FIRES  
  12. });  
  13.   
  14. $("div").click(function () {  
  15.   // THIS NEVER FIRES  
  16. });  

你可能会觉得这个例子看起来很别扭,没错,尽管如此,但有时这的确会发生,如果你的代码非常复杂,那么不同的widgets和plugin就有可能在同一个对象上添加事件,如果遇到这种情况,那你就很有必要理解和使用stopImmediatePropagation。

return false

只有当你同时需要preventDefault和stopPropagation,并且你的代码可以接受直到你的回调执行完成才停止执行浏览器的默 认行为,那你就可以使用”return false“。但我强烈建议你别在写给其它jQuery开发者的演示代码中使用这个方法,因为这会造成更多误用,只有在你确信非用不可的情况下再去使 用”return false“。

选择适当的位置

如果你使用了”return false“,它只会在你的回调函数执行结束才去取消浏览器的默认行为,但是使用e.preventDefault,我们有更多的选择,它可以随时停止浏览器执行默认动作,而不管你将它放在函数的哪个部分。

1. 开发阶段,你应该总是将它放在第一行。你最不想做的事情可能就是你正在调试将一个form改成ajax提交的时候,它却已经被按照老方法提交了。

2.产品阶段,如果你采用了渐进增强(progressive enhancement),那就把它放到回调的 结束位置,或者是逻辑终点,如果在一个普通页面采用渐进增强,那你就需要在服务器端考虑如果浏览器不支持JS时(或者被禁用时),对链接的click事件 和表单的提交事件的处理。这里的好处是,我们不考虑关闭JS的情况,只考虑支持js时的强狂,如果你的回调代码出错抛出了异常,让我们看看下面的代码:

  1. var data = {};  
  2. $("a").click(function (e) {  
  3.   e.preventDefault(); // cancel default behavior  
  4.   // Throws an error because `my` is undefined  
  5.   $("body").append(data.my.link);  
  6.   // The original link doesn't work AND the "cool"  
  7.   // JavaScript has broken. The user is left with NOTHING!  
  8. });  

现在,让我们看看同样的事件,把preventDefault调用放在底部的效果:

  1. >  
  2. var data = {};  
  3. $("a").click(function (e) {  
  4.   // Throws an error because `my` is undefined  
  5.   $("body").append(data.my.link);  
  6.   
  7.   // This line is never reached, and your website  
  8.   // falls back to using the `href` instead of this  
  9.   // "cool" broken JavaScript!  
  10.   e.preventDefault(); // cancel default behavior  
  11. });  

这对表单提交也同样有效,你可以更好的应对出错的情况,别指望你的代码一直正常工作,在发生错误时有正确的应对总胜过假设代码不会出错。

3. 在产品阶段,如果功能这设计JS,那就还应该放在第一行。

记住,不必非得是函数的第一行,但是越早越好,这里的原则是:如果函数的功能是通过JS实现的(不涉及服务端交互),那就没必要考虑兼容,在这种情 况下,添加在第一行可以防止URL中出现#字符,但显然,你还是应该尽可能多的增加些错误处理代码,以防止用户在出错时变得不知所措。

结论

我希望这篇文章传达的信息足够你在需要阻止浏览器执行默认行为时做出正确的选择。记住,只有当你真的明白你在做什么时,才使用”return false“,并确保你是在函数的正确位置调用了相应的代码。最后,尽可能保持代码的灵活性,尽量不要再用“return false”了!

 

from:http://qurtyy.blog.163.com/blog/static/5744368120130214373035/

分享到:
评论

相关推荐

    jquery监听鼠标滚轮事件+js监听滚轮事件

    注意,对于不支持`event.preventDefault()`的老版本IE,还需要设置`event.returnValue = false`来阻止默认行为。 总结,无论使用jQuery还是原生JavaScript,监听鼠标滚轮事件都能为网页带来更丰富的交互体验。在...

    JQuery事件e参数的方法preventDefault()取消默认行为

    它通过在事件处理函数中传递的事件对象`e`(event对象)来调用。当执行`e.preventDefault()`方法时,当前元素上的默认动作就不会被执行。比如在上面给出的HTML文档示例中,如果我们将`&lt;a&gt;`标签的链接地址指向百度,...

    js阻止默认浏览器行为与冒泡行为的实现代码.docx

    总的来说,理解和正确使用`event.preventDefault()`和`event.stopPropagation()`是JavaScript事件处理中的关键技能,它们能够帮助开发者实现更灵活的交互和控制。在实际项目中,根据具体需求选择合适的方法,确保...

    js阻止冒泡及jquery阻止事件冒泡示例介绍.docx

    ### JS与jQuery阻止事件冒泡详解 #### 一、引言 在Web开发中,事件处理是不可或缺的一部分,其中事件冒泡机制尤为重要。事件冒泡是指事件从最深层的节点开始,逐级向上传播至最顶层的节点。在某些情况下,我们希望...

    jquery的冒泡事件的阻止与允许(三种实现方法)

    return false 在事件的处理中,可以阻止默认事件和冒泡事件。 B:event.preventDefault()—&gt; In event handler ,prevent default event (allows bubbling) 。 event.preventDefault()在事件的处理

    学习jQuey中的return false

    总之,理解并恰当使用`return false`、`.preventDefault()` 和 `.stopPropagation()` 是编写高效、可控的jQuery事件处理代码的关键。根据实际情况选择合适的方法,可以有效地控制事件的执行流程,提高用户体验。

    jquery-django-form-源码.rar

    jQuery,作为一个轻量级的JavaScript库,简化了DOM操作、事件处理以及Ajax交互,极大地提高了前端开发效率。而Django,则是Python的一个高级Web框架,以其MVT(Model-View-Template)架构模式著称,提供了强大的后端...

    浅析return false的正确使用

    在JavaScript和jQuery中,`return false`是一个常见的用法,特别是在事件处理程序中,用于防止浏览器执行默认行为。然而,它的使用常常被误解,可能导致潜在的问题。本文将深入探讨何时以及如何正确地阻止浏览器的...

    jquery 事件冒泡的介绍以及如何阻止事件冒泡

    在上述示例中,如果在`hr_three`的点击事件处理函数中调用`event.preventDefault()`,则会依次弹出所有层级的警告,但不会跳转到百度。值得注意的是,即使`event.preventDefault()`阻止了超链接的跳转,事件仍然可以...

    jQuery阻止事件冒泡实例分析

    事件冒泡可以通过多种方式阻止,包括直接在事件处理函数中返回false、使用event对象的stopPropagation方法以及在旧版IE浏览器中使用cancelBubble属性。 下面是对这些阻止事件冒泡方法的具体分析: 1. 使用return ...

    java 仿google 搜索修正 jquery autocomplete

    为了增加键盘导航功能,我们需要扩展Autocomplete的事件处理,监听键盘的上下箭头键。以下是如何实现这个功能的代码示例: ```javascript $(function() { $("#searchInput").autocomplete({ source: "search_...

    jquery-1.12.4.zip

    3. **事件处理**:jQuery的事件处理更加便捷,如`click()`、`hover()`等,可以轻松绑定和解绑事件。例如,`$("button").click(function() {...})`用于监听按钮的点击事件。 4. **动画效果**:jQuery的`animate()`...

    jQuery鼠标右键弹出菜单选项.zip

    在前端开发中,jQuery库是广泛使用的一种JavaScript库,它简化了DOM操作、事件处理、动画效果和Ajax交互。在“jQuery鼠标右键弹出菜单选项”这个项目中,我们聚焦于利用jQuery来实现鼠标右键点击时显示一个自定义的...

    jQuery自定义鼠标右键弹出菜单代码

    在网页开发中,jQuery是一个非常流行的JavaScript库,它简化了DOM操作、事件处理以及动画创建。本示例中,我们将探讨如何使用jQuery实现一个自定义的鼠标右键弹出菜单功能,这与我们在电脑桌面上点击鼠标右键时看到...

    return false;和e.preventDefault();的区别

    Have you ever seen those two things (in the title) being used in jQuery?... return false; } That code would append the href attribute as text to the body every time a link was clicked but not actually go

    jquery验证

    jQuery是一个强大的JavaScript库,它极大地简化了JavaScript的DOM操作、事件处理和Ajax交互。在网页开发中,jQuery验证通常指的是使用jQuery库来实现表单验证,确保用户输入的数据符合预设的规则,从而提高用户体验...

    利用JQuery阻止事件冒泡

    除了使用event对象的方法外,JQuery的事件处理函数还提供了一个简写形式来阻止冒泡,即直接在事件处理函数的末尾返回false。这等同于调用了event.stopPropagation()和event.preventDefault()。例如: ```javascript...

    jQuery鼠标右键点击显示菜单代码

    在网页开发中,jQuery是一个非常流行的JavaScript库,它简化了DOM操作、事件处理以及动画制作等任务。在标题“jQuery鼠标右键点击显示菜单代码”中,我们要讨论的是如何利用jQuery来实现用户在页面上鼠标右键点击时...

Global site tag (gtag.js) - Google Analytics