jQuery事件:bind、delegate、on的区别
最近在AngularJS的开发中,遇到一个神奇的事情:我们用到livebox来预览评论列表中的图片, 然而评论列表是由Angular Resource动态载入的。不可思议的是,点击这些动态载入的图片仍然会触发lightbox的图片预览。 难道lightbox使用先进的MutationObserver技术监听了DOM的变化?观察lightbox源码才发现,原来只是jQuery的.on()
方法:
$('body').on('click', 'a[rel^=lightbox], ...', function(event){});
本文便来详解各种jQuery事件绑定方法:on
,bind
,delegate
,live
,unbind
,trigger
。 同时总结一下常用的jQuery事件技术:如何阻止事件冒泡、阻止浏览器默认行为、解绑事件处理函数、自定义事件。
什么是 jQuery 事件
jQuery事件是DOM事件的封装,同时支持自定义的扩展。在程序设计中,事件和代理有着相似的作用: 它们提供了一种机制,使得行为的实现方式和调用时机可以分离。
不谈jQuery,DOM本身就提供了一系列的javascript事件,例如click
,keyup
,submit
。 未实现相关业务逻辑,通常会为这些事件定义一系列的处理函数,处理函数定义了业务的实现方式,而浏览器知道这些业务的调用时机。 Javascript事件就是这样一种机制,使得行为的实现方式和调用时机可以动态地绑定。
jQuery事件是通过封装javascript事件来实现的,例如.keyup()
便是onkeyup
的封装:
.keyup()
: Bind an event handler to the “keyup” JavaScript event, or trigger that event on an element.
除了封装大多数的javascript事件,jQuery提供了统一的事件绑定和触发机制:
- 绑定事件:
bind
、on
、live
、delegate
、keyup(<function>)
; - 触发事件:
trigger('keyup')
、keyup()
; - 解绑事件:
unbind
、off
、die
、undelegate
。
事件绑定:bind
使用javascript绑定一个事件很简单,只需要在HTML中设置onxxx
属性, 并且在javascript中定义相关的处理函数便可以完成。
<div onclick="func()"></div>
<script>
function func(){
console.log('clicked!');
}
</script>
上述是基本的javascript事件处理方式,而jQuery提供了更加方便的方式:.bind()
函数。
.bind()
:Attach a handler to an event for the elements.
<div id='foo'></div>
<script>
$('#foo').click(function(){
console.log('clicked!');
});
</script>
.click(<function>)
等效于.bind('click', <function>)
。另外还可以通过unbind
来解绑事件:
$('#foo').unbind('click');
如果
unbind
参数为空,则解绑匹配元素的所有事件处理函数。 在我的理解中,我们还是不要off
,unbind
,die
吧。即使不谈效率,它们也使得软件更难理解了。 如果你感觉有需要,下面的.on()
应该会满足你~
.bind
将会给所有匹配的元素都绑定一次事件,当元素很多时性能会变差。 而且后来动态新增的元素不会被绑定。
事件冒泡与默认行为
在DOM中默认情况下,事件是会冒泡的,即同样的事件会沿着DOM树逐级触发。 有时这是我们不希望的行为,可以在事件处理函数中阻止它。
// 事件处理函数的第一个参数是一个事件对象
$('#foo').click(function(event){
event.stopPropagation();
// do sth.
});
浏览器对用户事件的默认行为是另一个需要考虑的事情,尤其是<a>
标签的click
事件。 当用户点击<a>
标签时,首先调用所有的事件处理函数,然后执行默认行为:页面跳转或者定位。 同样地,我们可以阻止它:
$('a').click(function(event){
event.preventDefault();
// do sth.
});
在实践中,我们常常让事件处理函数return false
来阻止冒泡和默认行为, 可以认为return false
做了三件事情:
stopPropagation()
;preventDefault()
;- 立即结束当前函数并返回。
$('a').click(function(event){
// do sth.
return false;
});
自定义事件
jQuery事件是基于DOM事件的,但jQuery提供了更加普遍的事件机制。 这使得我们可以方便地自定义事件,只需要给一个尚不存在的事件名即可:
<div id='foo'></div>
<script>
$('#foo').bind('fucked', function(){
console.log("I'm fucked.");
});
$('#foo').trigger('fucked');
</script>
这里定义了一个叫fucked
的事件并绑定了处理函数,然后使用trigger
来触发该事件。 在真实的场景中,通常用其他的事件来触发自定义事件:
var he = 'man';
$('#foo').click(function(){
if(he === 'man') $(this).trigger('fucked');
});
Delegate
.delegate
是另一种绑定事件的方式。它将事件处理函数绑定在指定的根元素上, 由于事件会冒泡,它用来处理指定的子元素上的事件。
.delegate()
:Attach a handler to one or more events for all elements that match the selector, now or in the future, based on a specific set of root elements.
<div id="root">
<a>Alice</a>
<a>Bob</a>
</div>
<script>
$('#root').delegate('a', 'click', function(){
console.log('clicked');
});
</script>
它的使用方式比bind
稍显复杂,但它的功能非常强大:
- 自动绑定动态添加的元素。因为事件处理函数绑定在
#root
上,新加的子元素事件也会冒泡到#root
。 - 性能好于
.bind()
。只绑定一个事件处理函数,绑定速度相当快。
如果你在使用AngularJS等动态操作DOM的工具,那么
.delegate()
将会非常实用,它能对新增的DOM元素自动绑定。
On
事实上,.on()
才是jQuery事件的提供者。其他的事件绑定方法都是通过.on()
来实现的,请看jQuery1.8.2的源码:
bind: function( types, data, fn ) {
return this.on( types, null, data, fn );
},
unbind: function( types, fn ) {
return this.off( types, null, fn );
},
live: function( types, data, fn ) {
jQuery( this.context ).on( types, this.selector, data, fn );
return this;
},
die: function( types, fn ) {
jQuery( this.context ).off( types, this.selector || "**", fn );
return this;
},
delegate: function( selector, types, data, fn ) {
return this.on( types, selector, data, fn );
},
undelegate: function( selector, types, fn ) {
return arguments.length === 1 ? this.off( selector, "**" ) : this.off( types, selector || "**", fn );
},
既然.on
是最通用的jQuery事件机制,那么上述的所有例子都可以用.on()
来实现:
// bind
$( "#foo" ).bind( "click", function( e ) {} );
$( "#foo" ).on( "click", function( e ) {} );
// delegate
$( "#root" ).delegate( "a", "click", function( e ) {} );
$( "#root" ).on( "click", "a", function( e ) {} );
我们看到上面还有一个
.live()
方法,它与delegate
是类似的, 不过它强制指定了root
是document
(即this.context
),因而性能略差。 自jQuery1.7起已经不推荐使用了。参见: https://api.jquery.com/category/deprecated/deprecated-1.7/
jQuery事件:bind、delegate、on的区别的更多相关文章
- Jquery中bind和live的区别
Jquery中绑定事件有三种方法:以click事件为例 (1)target.click(function(){}); (2)target.bind("click",function ...
- jQuery 事件 - bind() 方法
定义和用法 bind() 方法为被选元素添加一个或多个事件处理程序,并规定事件发生时运行的函数. 实例1(一个事件) 记得把js引用地址换掉 当点击鼠标时,隐藏或显示 p 元素: <html&g ...
- jquery中bind和on的区别
1.首先,来看看bind和on的语法. bind的用法: $('a').bind('click',[data],function(){}) 其事件的绑定者是固定的,就是a,第一个参数是事件,第二个参数 ...
- jQuery的.bind()、.live()和.delegate()的区别
参考:http://kb.cnblogs.com/page/94469/ 摘要:jQuery的.bind()..live()和.delegate()之间的区别并非总是那么明显的,然而,如果我们对所有的 ...
- jQuery的.bind()、.live()和.delegate()之间区别
摘要:jQuery的.bind()..live()和.delegate()之间的区别并非总是那么明显的,然而,如果我们对所有的不同之处都有清晰的理解的话,那么这将会有助于我们编写出更加简洁的代码,以及 ...
- jQuery的.bind()、.live()和.delegate(),on之间区别
基本要素 51CTO推荐专题:jQuery从入门到精通 DOM树 首先,可视化一个HMTL文档的DOM树是很有帮助的.一个简单的HTML页面看起来就像是这个样子: 事件冒泡(又称事件传播) 当我们点击 ...
- Jquery中bind和live.one,delegate的区别
Jquery中绑定事件有三种方法:以click事件为例 (1)target.click(function(){}); (2)target.bind("click",function ...
- jQuery事件绑定—on()、bind()与delegate()
啃了一段日子的js相关了,学的过程中发现在jQuery中绑定事件时,有人用bind(),有人用on(),有人用delegate(),还有人用live(),看代码的时候觉得都实现功能了也就掀过去了,只是 ...
- Jquery中.bind()、.live()、.delegate()和.on()之间的区别详解
简介 最近了解到很多网页开发者对jquery中的 .bind() .live() .delegate() 和 .on() 方法存在很多的疑惑.这些疑惑通常是关于它们之间真正的区别是什么啊,什么时候该使 ...
随机推荐
- sourceinsight 工程和源码不在同一个盘符下
建立sourceinsight的时候,si工程可以和项目源码不在同一个盘下面,即si工程在D盘下,而阅读的源码在E盘下. 方法步骤如下: 下看一下目录结构: Y:\work\Hi3521\Hi3521 ...
- [Android Pro] android root权限破解分析
许 多机友新购来的Android机器没有破解过Root权限,无法使用一些需要高权限的软件,以及进行一些高权限的操作,其实破解手机Root权限是比较简 单及安全的,破解Root权限的原理就是在手机的/s ...
- Servlet介绍以及简单实例
一.背景介绍: HTTP:超文本传输协议(HTTP,HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议.设计HTTP最初的目的是为了提供一种发布和接收 HTM ...
- Kotlin(2): 优雅地扩展类的方法和属性
欢迎Follow我的GitHub, 关注我的CSDN. 个人博客: http://www.wangchenlong.org/, 最新内容. Kotlin由JetBrains公司推出, 是兼容Java的 ...
- uva539 卡坦岛 简单回溯!
继续回溯搞起! 开始想复杂了,用了好多数组判断节点的度.边是否已经走过,结果导致超时了,后来简化成如下版本,走过的标志不需要另辟vis数组,只要将map[i][j]和map[j][i]赋值0即可. # ...
- 当Intellij IDEA 遇到 Mac
当Intellij IDEA 遇到 Mac alt+insert变为control+N
- 表结构设计器(EZDML)1.98版公布
表结构设计器(EZDML)是一个免费的数据库建表的小软件,可高速的进行数据库表结构设计.建立数据模型,能迅速生成代码模板.简单界面和字典文档,支持脚本编程. 新版本号大概有下面改进: 1.修复了部 ...
- IIS7.5 取消301重定向
今天想把一个域名解析到对应的www的域名,添加了301重定向. 结果域名解析那里是把www解析到了对应的域名,结果就是重定向循环.立即删除了IIS中重定向,结果还是无法解决. 以为是有缓存,重 ...
- 查找文件工具find
与locate.whereis命令相比,find具有本质的区别: 首先,find是从指定的位置进行遍历查找(可以理解为对文件和目录进行逐一查找). 其次,find可以查找具有某一类特征的文件(例如查找 ...
- NSURLConnection经常使用的代理方法
NSURLConnection的代理Protocol定义有三类:NSURLConnectionDelegate.NSURLConnectionDataDelegate和NSURLConnectionD ...