深入理解JS的事件绑定、事件流模型


Posted in Javascript onMay 13, 2018

 一、JS事件

(一)JS事件分类

1.鼠标事件:

click/dbclick/mouseover/mouseout

2.HTML事件:

 onload/onunload/onsubmit/onresize/onchange/onfoucs/onscroll

3.键盘事件:

 keydown:键盘按下时触发
 keypress:键盘按下并抬起的瞬间触发。
 keyup:键盘抬起触发

[注意事项]

①执行顺序:keydown keypress keyup
②keypress只能捕获数字,字母,符号键,而不能捕获功能键。
③长按时循环执行keydown--keypress
④有keydown,并不一定有keyup,当长按时焦点失去,将不再触发keyup
⑤keypress区分大小写,keydown,kewup不区分。

4.事件因子:

当触发一个事件时,该事件将向事件所调用的函数中,默认传入一个参数,

这个参数就是一个事件因子,包含了该事件的各种详细信息。

document.onkeydown=function(e){
 console.log(e);
 }
document.onkeydown=function(){
console.log(window.event);
}
//兼容浏览器的写法:
document.onkeydown=function(e){
e==e||Window.event;
var Code=e.keyCode||e.which||e.charCode;
if(code==13){
//回车
}
}

5.如何确定键盘按键?

①再出发的函数中,接收事件因子e。
②可以使用e.key直接去到按下的按键字符(不推荐使用)。
③一次可以使用keyCode/which/charCode取到按键的ASCII码值。

(兼容各种浏览器的写法)

var Code=e.keyCode||e.which||e.charCode;

//判断组合键
var isAlt=0,isEnt=0;
document.onkeyup=function(e){
if(e.keyCode==18){
isAlt=1;
}      
if(e.keyCode==13){
isEnt=1;
}    
if(isAlt==1&&isEnt==1){
alert("同时按下Alt和Enter键");
}
}

document.onkeyup=function(){
console.log("keyup");
}
document.onkeypress=function(){
console.log("keypress");
}
document.onkeydown=function(){
console.log("keydown");
}
document.onkeypress=function(){
console.log(window.event);
}
//判断是否按下了回车键
document.onkeydown=function(e){
var code=e.keyCode;
if(code==13){
alert("你输入的是回车键");
}
}

二、事件绑定模型

(一)DOM0事件模型

绑定注意事项:

①使用window.onload加载完成后进行绑定。

window.onload =function(){//事件}

②放在body后面进行绑定。

//body内容
<body>
  <button onclick="func()">内联模型绑定</button>
  <button id="btn1">哈哈哈哈</button>
  <button id="btn2">DOM2模型绑定</button>
  <button id="btn3">取消DOM2</button>
</body>

1.内联模型(行内绑定):将函数名直接作为html标签中属性的属性值。

<button onclick="func()">内联模型绑定</button>

 缺点:不符合w3c中关于内容与 行为分离的基本规范。

2.脚本模型(动态绑定):通过在JS中选中某个节点,然后给节点添加onclick属性。

document.getElementById("btn1")=function(){}

 优点:符合w3c中关于内容与行为分离的基本规范,实现html与js的分离。
 缺点:同一个节点只能添加一次同类型事件,如果添加多次,最后一个生效。

document.getElementById("btn1").onclick=function(){
  alert(1234);  
}
document.getElementById("btn1").onclick=function(){
  alert(234);  
}//重复的只能出现最近的一次

3.DOM0共有缺点:通过DOM0绑定的事件,一旦绑定将无法取消。

document.getElementById("btn3").onclick=function(){//不能取消匿名函数
  if(btn.detachEvent){
    btn.detachEvent("onclick",func1);
  }else{
    btn.removeEventListener("click",func1);
  }
    alert("取消DOM2");
}

(二)DOM2事件模型

1.添加DOM2事件绑定:

 ①IE8之前,使用.attachEvent("onclick",函数);
 ②IE8之后,使用.addEventListener("click",函数,true/false);
 参数三:false(默认)表示事件冒泡,传入true表示事件捕获。
 ③兼容所有浏览器的处理方式:

var btn=document.getElementById("btn1");
 if(btn.attachEvent){
 btn.attachEvent("onclick",func1);//事件,事件需要执行的函数IE8可以
 }else{
 btn.attachEventListener("click",func1);
 }

2.DOM2绑定的优点:

 ①同一个节点,可以使用DOM2绑定多个同类型事件。
 ②使用DOM2绑定的事件,可以有专门的函数进行取消。
3.取消事件绑定:
 ①使用attachEvent绑定,要用detachevent取消。
 ②使用attachEventListener绑定,要用removeEventListenter取消。
 注意:如果DOM2绑定的事件,需要取消,则绑定事件时,回调函数必须是函数名,
 而不能是匿名函数,因为取消事件时,取消传入函数名进行取消。

三、JS事件流模型

(一)JS中的事件流模型

1. 事件冒泡(fasle/不写):当触发一个节点的事件是,会从当前节点开始,依次触发其祖先节点的同类型事件,直到DOM根节点。
2. 事件捕获(true):当初发一个节点的事件时,会从DOM根节点开始,依次触发其祖先节点的同类型事件,直到当前节点自身。
3. 什么时候事件冒泡?什么时候事件捕获?
 ① 当使用addEventListener绑定事件,第三个参数传为true时表示事件捕获;
 ② 除此之外的所有事件绑定均为事件冒泡。

4. 阻止事件冒泡:

 ① IE10之前,e.cancelBubble = true;
 ② IE10之后,e.stopPropagation();

5. 阻止默认事件:

 ① IE10之前:e.returnValue = false;
 ② IE10之后:e.preventDefault();

//css
#div1{
  width: 300px;;
  height: 300px;
  background-color: powderblue;
}
#div2{
  width: 200px;
  height: 200px;
  background-color: deeppink;
}
#div3{
  width: 100px;
  height: 100px;
  background-color:#A9A9A9;
}

//html
<div id="div1">
  <div id="div2">
    <div id="div3"></div>
  </div>
</div>
<a href="01-事件笔记.html" rel="external nofollow" onclick="func()">超链接</a>
div1.addEventListener("click",function(){
  console.log("div1 click");
},false);
div2.addEventListener("click",function(){
  console.log("div2 click");
},false);
div3.addEventListener("click",function(){ //原来的顺序是:3-->2-->1。
//  myParagraphEventHandler(); //截获事件流后,只触发3.但是从2开始依然会冒泡;
  console.log("div3 click");
},false);

结果(事件冒泡)(由小到大div3-》div2-》div1):

深入理解JS的事件绑定、事件流模型

div1.addEventListener("click",function(){
  console.log("div1 click");
},true);
div2.addEventListener("click",function(){
  console.log("div2 click");
},true);
div3.addEventListener("click",function(){ 
//  myParagraphEventHandler(); //截获事件流后,只触发3.但是从2开始依然会冒泡;
  console.log("div3 click");
},true);

结果(事件捕获)(由小到大div3-》div2-》div1):

深入理解JS的事件绑定、事件流模型

//依然遵循事件冒泡
document.onclick=function(){
  console.log("document click")
}
//截获事件流阻止事件冒泡
function myParagraphEventHandler(e) {
   e = e || window.event;
  if (e.stopPropagation) {
     e.stopPropagation(); //IE10以后 
  } else {
     e.cancelBubble = true; //IE10之前
   }
}
//截获事件流阻止事件冒泡
function myParagraphEventHandler(e) {
   e = e || window.event;
  if (e.stopPropagation) {
     e.stopPropagation(); //IE10以后 
  } else {
     e.cancelBubble = true; //IE10之前
   }
}
//阻止默认事件
function eventHandler(e) {
  e = e || window.event;
// 防止默认行为 
   if (e.preventDefault) {
     e.preventDefault(); //IE10之后
  } else {
     e.returnValue = false; //IE10之前  
  }
}

总结

以上所述是小编给大家介绍的JS的事件绑定、事件流模型,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

Javascript 相关文章推荐
javascript:window.open弹出窗口的位置问题
Mar 18 Javascript
javascript基于DOM实现权限选择实例分析
May 14 Javascript
jQuery实现延迟跳转的方法
Jun 05 Javascript
require.js的用法详解
Oct 20 Javascript
js显示动态时间的方法详解
Aug 20 Javascript
用jmSlip编写移动端顶部日历选择控件
Oct 24 Javascript
js数组去重的hash方法
Dec 22 Javascript
推荐10款扩展Web表单的JS插件
Dec 25 Javascript
详解如何理解vue的key属性
Apr 14 Javascript
vue实现商品列表的添加删除实例讲解
May 14 Javascript
原生js实现自定义消息提示框
Nov 19 Javascript
JavaScript的Set数据结构详解
Feb 18 Javascript
在ES5与ES6环境下处理函数默认参数的实现方法
May 13 #Javascript
vue中的$emit 与$on父子组件与兄弟组件的之间通信方式
May 13 #Javascript
node 命令方式启动修改端口的方法
May 12 #Javascript
安装Node.js并启动本地服务的操作教程
May 12 #Javascript
Node.js中,在cmd界面,进入退出Node.js运行环境的方法
May 12 #Javascript
Vue拖拽组件开发实例详解
May 11 #Javascript
node前端模板引擎Jade之标签的基本写法
May 11 #Javascript
You might like
php的字符串用法小结
2010/06/08 PHP
浅析PHP中的 inet_pton 网络函数
2019/12/16 PHP
运用Windows XP附带的Msicuu.exe、Msizap.exe来彻底卸载顽固程序
2007/04/21 Javascript
关于js拖拽上传 [一个拖拽上传修改头像的流程]
2011/07/13 Javascript
javascript笔记 String类replace函数的一些事
2011/09/22 Javascript
javascript中的delete使用详解
2013/04/11 Javascript
JQuery触发事件例如click
2013/09/11 Javascript
php跨域调用json的例子
2013/11/13 Javascript
jQuery实现每隔几条元素增加1条线的方法
2016/06/27 Javascript
BootStrap 动态添加验证项和取消验证项的实现方法
2016/09/28 Javascript
PHP+jquery+ajax实现分页
2016/12/09 Javascript
js实现3d悬浮效果
2017/02/16 Javascript
vue中使用cropperjs的方法
2018/03/01 Javascript
vue 2.1.3 实时显示当前时间,每秒更新的方法
2018/09/16 Javascript
使用vue制作滑动标签
2019/09/21 Javascript
node读写Excel操作实例分析
2019/11/06 Javascript
简单了解JavaScript sort方法
2019/11/25 Javascript
python基于mysql实现的简单队列以及跨进程锁实例详解
2014/07/07 Python
python3实现磁盘空间监控
2018/06/21 Python
python pygame模块编写飞机大战
2018/11/20 Python
python抖音表白程序源代码
2019/04/07 Python
解决keras backend 越跑越慢问题
2020/06/18 Python
socket.io 和canvas 实现的共享画板功能
2019/05/22 HTML / CSS
英国泽西岛植物:Jersey Plants Direct
2019/08/07 全球购物
应届毕业生简历自我评价
2014/01/31 职场文书
自主招生教师推荐信
2014/05/10 职场文书
小学校长竞聘演讲稿
2014/05/16 职场文书
元旦晚会活动总结
2014/07/09 职场文书
企业挂职心得体会
2014/09/10 职场文书
2014年党员自我评议(5篇)
2014/09/12 职场文书
出售房屋委托书范本
2014/09/24 职场文书
2014小学数学教师个人工作总结
2014/12/18 职场文书
高三复习计划
2015/01/19 职场文书
校运会宣传稿大全
2015/07/23 职场文书
安全教育日主题班会
2015/08/13 职场文书
学生会部长竞选稿
2015/11/19 职场文书