浏览器JS事件触发机制

更新日期: 2019-01-30阅读: 3.8k标签: 机制

原理

事件捕获

由网景最先提出,事件会从最外层开始发生,直到最具体的元素,也就是说假如父元素与子元素都绑定有点击事件,又互相重叠,那么先出发的会是父元素的事件,然后再传递到子元素。


事件冒泡

由微软提出,事件会从最内从的元素开始发生,再向外传播,正好与事件捕获相反。

这两个概念都是为了解决页面中事件流的发生顺序,w3c采取了折中的办法,制定了统一的标准:先捕获再冒泡。


addEventListener

addEventListen(event, function, useCapture)添加事件的第三个参数默认值为false,即默认使用事件冒泡,若为true则使用事件捕获的机制,以下为示例:

clipboard.png

当我们使用默认事件注册机制的时候,点击child元素时,会先后输出child,container,这就是事件冒泡机制:

container.addEventListener('click', () => console.log('container'))
child.addEventListener('click', () => console.log('child'))

将第三个参数设为true,点击child元素时,输出顺序就会变为container、child:

container.addEventListener('click', () => console.log('container'), true)
child.addEventListener('click', () => console.log('child'), true)

假若你希望点击child只打印child,不触发container的事件,我们就需要用到stopPropagation,它阻止捕获和冒泡阶段中当前事件的进一步传播:

container.addEventListener('click', () => console.log('container'))
child.addEventListener('click', (e) => {
  e.stopPropagation(); 
  console.log('child');
})

若你想要点击child只打印container,就需要进行捕获事件,并在container事件触发时进行阻止捕获:

container.addEventListener('click', (e) => {
  e.stopPropagation(); 
  console.log('container')
}, true)
child.addEventListener('click', () => console.log('child'), true)

与stopPropagation,还有一种事件方式叫做preventDefault,它的作用不是用于阻止冒泡,而是阻止浏览器默认行为,如a标签跳转,表单提交等。


事件委托

事件委托,又称为事件代理,通俗来讲是将元素的事件函数处理交由其他对象处理。它允许您避免向特定节点添加事件侦听器,我们这里所谈论的事件委托,与冒泡捕获流程相关,因此事件委托在此场景指的是子对象的处理事件交由父对象处理。

当你需要为一个动态的列表元素添加click事件时,若直接对列表项绑定事件,可能会重复注册很多个事件监听器。为了解决上述问题,我们可以利用事件委托的思想,在父级注册一个事件监听器,统一进行子元素的事件处理。

为了更好的说明,在这里举一个示例:

<ul class="list">
  <li class="item" data-id="1">1</li>
  <li class="item" data-id="2">2</li>
  <li class="item" data-id="3">3</li>
  <li class="item" data-id="4">4</li>
  <li class="item" data-id="5">5</li>
</ul>

为了给每一个li绑定上事件,若采用通常的方法,可能我们需要这样写代码

document.querySelectorAll('.item').forEach(item => {
  item.onclick = e => handleClick(e.target.dataset.id)
})

但是列表数据可能会动态的变化,这样我们就避免不了动态的去注册事件,数量还可能很多,那有没有什么一劳永逸的方法呢?当然,让我们使用事件委托的方法,也就是将点击事件绑定到ul元素上:

document.querySelector('.list').onclick = e => {
  if (e.target.matches('li.item')) {
    handleClick(e.target.dataset.id)
  }
}

值得注意的是,event.target代表事件起源目标的引用,若要寻找当前注册事件对象的引用,请用event.currentTarget。


链接: https://www.fly63.com/article/detial/2173

浅析前端页面渲染机制

作为一个前端开发,最常见的运行环境应该是浏览器吧,为了更好的通过浏览器把优秀的产品带给用户,也为了更好的发展自己的前端职业之路,有必要了解从我们在浏览器地址栏输入网址到看到页面这期间浏览器是如何进行工作的

这一次,彻底弄懂 JavaScript 执行机制

javascript是一门单线程语言,Event Loop是javascript的执行机制.牢牢把握两个基本点,以认真学习javascript为中心,早日实现成为前端高手的伟大梦想!

创建js hook钩子_js中的钩子机制与实现

钩子机制也叫hook机制,或者你可以把它理解成一种匹配机制,就是我们在代码中设置一些钩子,然后程序执行时自动去匹配这些钩子;这样做的好处就是提高了程序的执行效率,减少了if else 的使用同事优化代码结构

小程序的更新机制_如何实现强制更新?

在讲小程序的更新机制之前,我们需要先了解小程序的2种启动模式,分别为:冷启动和热启动。小程序不同的启动方式,对应的更新情况不不一样的。无论冷启动,还是热启动。小程序都不会马上更新的,如果我们需要强制更新,需要如何实现呢?

基于JWT的Token认证机制实现及安全问题

JSON Web Token(JWT)是一个非常轻巧的规范。这个规范允许我们使用JWT在用户和服务器之间传递安全可靠的信息。其JWT的组成:一个JWT实际上就是一个字符串,它由三部分组成,头部、载荷与签名。

web前端-JavaScript的运行机制

本文介绍JavaScript运行机制,JavaScript语言的一大特点就是单线程,也就是说,同一个时间只能做一件事。单线程就意味着,所有任务需要排队,前一个任务结束,才会执行后一个任务。

轮询机制解决后端任务回调问题

现在有一个需求,前端有一个按钮,点击以后会调用后端一个接口,这个接口会根据用户的筛选条件去hadoop上跑任务,将图片的base64转为img然后打包成zip,生成一个下载连接返回给前端,弹出下载框。hadoop上的这个任务耗时比较久

JavaScript预解释是一种毫无节操的机制

js代码执行之前,浏览器首先会默认的把所有带var和function的进行提前的声明或者定义:1.理解声明和定义、2.对于带var和function关键字的在预解释的时候操作不一样的、3.预解释只发生在当前的作用域下

js对代码解析机制

脚本执行js引擎都做了什么呢?1.语法分析 2.预编译 3.解释执行。在执行代码前,还有两个步骤;语法分析很简单,就是引擎检查你的代码有没有什么低级的语法错误 ,查找全局变量声明(包括隐式全局变量声明,省略var声明),变量名作全局对象的属性,值为undefined

web认证机制

以前对认证这方面的认识一直不太深刻,不清楚为什么需要token这种认证,为什么不简单使用session存储用户登录信息等。最近读了几篇大牛的博客才对认证机制方面有了进一步了解。

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!