jQuery的.bind()、.live()和.delegate()之间的区别

标签: jquery bind live | 发表时间:2011-03-16 17:13 | 作者:Elaine.Ye Allen
出处:http://www.yeeyan.org

原作者:
来源The Difference Between jQuery’s bind(), live(), and delegate() - Alfa Jango Blog
译者Elaine.Ye

原文:The Difference Between jQuery’s .bind(), .live(), and .delegate()

作者:Steve Schwartz

时间:2011-02-01

出处:http://www.alfajango.com/blog/the-difference-between-jquerys-bind-live-and-delegate/

    .bind().live().delegate()之间的区别并非总是那么明显的,然而,如果我们对所有的不同之处都有清晰的理解的话,那么这将会有助于我们编写出更加简洁的代码,以及防止在交互应用中弹出错误。

   

 基本要素

DOM

首先,可视化一个HMTL文档的DOM树是很有帮助的。一个简单的HTML页面看起来就像是这个样子:

事件冒泡(又称事件传播)

    当我们点击一个链接时,其触发了链接元素的单击事件,该事件则引发任何我们已绑定到该元素的单击事件上的函数的执行。

$('a').bind('click', function() { alert("That tickles!") });

因此一个单击操作会触发alert函数的执行。

click事件接着会向树的根方向传播,广播到父元素,然后接着是每个祖先元素,只要是它的某个后代元素上的单击事件被触发,事件就会传给它。

在操纵DOM的语境中,document是根节点。

现在我们可以较容易地说明.bind().live().delegate()的不同之处了。

.bind()

$('a').bind('click', function() { alert("That tickles!") });

这是最简单的绑定方法了。JQuery扫描文档找出所有的$(‘a’)元素,并把alert函数绑定到每个元素的click事件上。

.live()

         $('a').live('click', function() { alert("That tickles!") });

    JQueryalert函数绑定到$(document)元素上,并使用’click’’a’作为参数。任何时候只要有事件冒泡到document节点上,它就查看该事件是否是一个click事件,以及该事件的目标元素与’a’这一CSS选择器是否匹配,如果都是的话,则执行函数。

live方法还可以被绑定到具体的元素(或“context”)而不是document上,像这样:

$('a', $('#container')[0]).live(...);

.delegate()

$('#container').delegate('a', 'click', function() { alert("That tickles!") });

JQuery扫描文档查找$(‘#container’),并使用click事件和’a’这一CSS选择器作为参数把alert函数绑定到$(‘#container’)上。任何时候只要有事件冒泡到$(‘#container’)上,它就查看该事件是否是click事件,以及该事件的目标元素是否与CCS选择器相匹配。如果两种检查的结果都为真的话,它就执行函数。

可以注意到,这一过程与.live()类似,但是其把处理程序绑定到具体的元素而非document这一根上。精明的JS’er们可能会做出这样的结论,即$('a').live() == $(document).delegate('a'),是这样吗?嗯,不,不完全是。

为什么.delegate()要比.live()好用

基于几个原因,人们通常更愿意选用jQuerydelegate方法而不是live方法。考虑下面的例子:

$('a').live('click', function() { blah() });

// 或者

$(document).delegate('a', 'click', function() { blah() });

速度

后者实际上要快过前者,因为前者首先要扫描整个的文档查找所有的$(‘a’)元素,把它们存成jQuery对象。尽管live函数仅需要把’a’作为串参数传递以用做之后的判断,但是$()函数并未“知道”被链接的方法将会是.live()

而另一方面,delegate方法仅需要查找并存储$(document)元素。

一种寻求避开这一问题的方法是调用在$(document).ready()之外绑定的live,这样它就会立即执行。在这种方式下,其会在DOM获得填充之前运行,因此就不会查找元素或是创建jQuery对象了。

灵活性和链能力

live函数也挺令人费解的。想想看,它被链到$(‘a’)对象集上,但其实际上是在$(document)对象上发生作用。由于这个原因,它能够试图以一种吓死人的方式来把方法链到自身上。实际上,我想说的是,以$.live(‘a’,…)这一形式作为一种全局性的jQuery方法,live方法会更具意义一些。

仅支持CSS选择器

最后一点,live方法有一个非常大的缺点,那就是它仅能针对直接的CSS选择器做操作,这使得它变得非常的不灵活。

欲了解更多关于CSS选择器的缺点,请参阅Exploring jQuery .live() and .die()一文。

更新:感谢Hacker News上的pedalpete后面评论中的Ellsass提醒我加入接下来的这一节内容。

为什么选择.live().delegate()而不是.bind()

毕竟,bind看起来似乎更加的明确和直接,难道不是吗?嗯,有两个原因让我们更愿意选择delegatelive而不是bind

Ÿ           为了把处理程序附加到可能还未存在于DOM中的DOM元素之上。因为bind是直接把处理程序绑定到各个元素上,它不能把处理程序绑定到还未存在于页面中的元素之上。

Ÿ           如果你运行了$(‘a’).bind(…),而后新的链接经由AJAX加入到了页面中,则你的bind处理程序对于这些新加入的链接来说是无效的。而另一方面livedelegate则是被绑定到另一个祖先节点上,因此其对于任何目前或是将来存在于该祖先元素之内的元素都是有效的。

Ÿ           或者为了把处理程序附加到单个元素上或是一小组元素之上,监听后代元素上的事件而不是循环遍历并把同一个函数逐个附加到DOM中的100个元素上。把处理程序附加到一个(或是一小组)祖先元素上而不是直接把处理程序附加到页面中的所有元素上,这种做法带来了性能上的好处。

停止传播

最后一个我想做的提醒与事件传播有关。通常情况下,我们可以通过使用这样的事件方法来终止处理函数的执行:

$('a').bind('click', function(e) {

  e.preventDefault();

  // 或者

  e.stopPropagation();

});

不过,当我们使用live或是delegate方法的时候,处理函数实际上并没有在运行,需要等到事件冒泡到处理程序实际绑定的元素上时函数才会运行。而到此时为止,我们的其他的来自.bind()的处理函数早已运行了。

 

添加新评论

相关文章:

  用CD创建一个Ubuntu的live USB

  我们所谓的生活

  我们所熟知的生活

  我们熟知的生活

  我们眼中的生活

相关 [jquery bind live] 推荐:

jQuery的.bind()、.live()和.delegate()之间的区别

- Allen - 译言-电脑/网络/数码科技
来源The Difference Between jQuery’s bind(), live(), and delegate() - Alfa Jango Blog. 原文:The Difference Between jQuery’s .bind(), .live(), and .delegate().

微软关闭Windows Live Spaces

- 柑桔 - 月光博客
  微软今天发布消息称,微软即将关闭已经运营了四年的Windows Live Spaces服务,并在未来六个月中将用户转移至WordPress.com博客平台.   微软和Wordpress共同宣布了这一消息,目前Windows Live Spaces拥有3000万用户,而WordPress拥有2600万用户.

Sony Ericsson Live With Walkman 评测

- 品味视界 - Engadget 中国版
刚刚一个九月,推出的手机不多,值得让我们去详细试玩评测的更少. 不过,作为 Walkman 系列中规格比较象样的一部手机,Live With Walkman 我们还是一定要评测的,虽然整体和其他 Xperia 系列手机的分别不大,但加上 Walkman 的标记,难免会让人有所期望. 到底 Live With Walkman 能不能让 Walkman 手机复活,以一部手机而言它又是否出色.

Windows Live支持XMPP协议

- 康爷 - Solidot
微软在BUILD会议上宣布Windows Live Messenger将支持开放标准网络即时消息协议XMPP. Google Talk、Facebook聊天服务和IBM Lotus Sametime都等支持XMPP. 微软表示,Windows Live Messenger将提供XMPP接口,允许用户将Messenger整合到基于Web、桌面和移动IM产品中.

基于云的Live Debian系统

- greenar - Solidot
goofegg 写道 "中国科技大学的学生建造了一套校内使用的基于云的Live Debian系统. 文章称:“借助于科大PXE服务(校外无法访问),我们正在定制一个‘云存储’Linux 系统. 此前,科大校园内通过PXE启动可以进入一些Live Linux系统,但在这些系统中的 所有修改都仅存在于内存中,关机后就丢失了.

Linkin Park - Rolling In The Deep(live中英字幕)

- Alen - 音动我心 – Mtime时光网

Xbox Live 确认将会登陆 Windows 8

- ts - Engadget 中国版
看过 Windows 8 新功能总整理后,感觉上缺了一些东西. 没错了,就是在 Windows Phone 7 上也能看见的 Xbox Live. 数天前,微软全球市场及互动娱乐部门的副执行长 Mike Delman 曾经在 The Seattle Times 上说过「Xbox Live 已在 Windows Phone 上取得成就」,「Xbox Live 将会内置于 PC 中.

Nginx+ffmpeg搭建Apple Http Live Streaming笔记

- - 移动开发 - ITeye博客
起始Nginx来搭建HLS步骤非常少. 安装好Nginx,然后跑起来. 把切片好的视频和m3u8文件放到部署目录,直接访问就可以了. 网上国内国外的找了好多博客. 这里一定注意一点,不要用VLC播放器来测试,最好用iPad或者iPhone,再么用Safari 开发模式下模拟iPad、iPhone的浏览器模式播放.

Redis服务器监控工具redis-live

- - 企业架构 - ITeye博客
Redis服务器监控工具redis-live. 413 views     comments 暂无评论. 目前来说,越来越多的使用多了NOSQL的业务,但是这方面的监控缺不多. 今天给大家介绍几个专业监控redis服务的工具,便于大家进行redis性能分析. 这个工具是用ruby语言写的,ruby是小鬼子弄出来的,个人真心觉得比较难用.

JQuery 选择器

- - CSDN博客Web前端推荐文章
}

点击我

.    像上面这样把JavaSript代码和HTML代码混杂在一起的做法同样也非常不妥,因为它并没有将网页内容和行为分离,所以才有JQuery选择器的学习.

点击我

. //给class为demo的元素添加行为.