一篇文章搞懂前端事件监听

简介: 一篇文章搞懂前端事件监听

认识事件监听

  • 前面我们讲到了JavaScript脚本和浏览器之间交互时,浏览器给我们提供的BOM、DOM等一些对象模型。
  • 事实上还有一种需要和浏览器经常交互的事情就是事件监听:
  • 浏览器在某个时刻可能会发生一些事件,比如鼠标点击、移动、滚动、获取、失去焦点、输入内容等等一系列 的事件;
  • 我们需要以某种方式(代码)来对其进行响应,进行一些事件的处理;
  • Web当中,事件在浏览器窗口中被触发,并且通过绑定到某些元素上或者浏览器窗口本身,那么我们就可以 给这些元素或者window窗口来绑定事件的处理程序,来对事件进行监听。
  • 如何进行事件监听呢?
  • 事件监听方式一:在script中直接监听;
  • 事件监听方式二:通过元素的on来监听事件;
  • 事件监听方式三:通过EventTarget中的addEventListener来监听;

事件监听的三种方式


function divClick() {
  console.log("div元素被点击2")
}
const divEl = document.querySelector(".box")
// DOM0
divEl.onclick = function() {
  console.log("div元素被点击3")
}
// DOM2
divEl.addEventListener("click", () => {
  console.log("div元素被点击4")
})
divEl.addEventListener("click", () => {
  console.log("div元素被点击5")
})
divEl.addEventListener("click", () => {
  console.log("div元素被点击6")
})

认识事件流的由来

  • 事实上对于事件有一个概念叫做事件流,为什么会产生事件流呢?
  • 我们可以想到一个问题:当我们在浏览器上对着一个元素点击时,你点击的不仅仅是这个元素本身;
  • 这是因为我们的HTML元素是存在父子元素叠加层级的;
  • 比如一个span元素是放在div元素上的,div元素是放在body元素上的,body元素是放在html元素上的;

代码演示


<body>
    <div class="box">
        <span class="span">我是span元素<span>
    </div>
</body>
document.body.addEventListener("click",()=>{
    console.log("body被点击")
})
const divEl = document.querySelector(".box")
const spanEl = document.querySelector(".span")
divEl.addEventListener("click",()=>{
   console.log("div被点击了")
})
spanEl.addEventListener("click",()=>{
   console.log("span被点击了")
})

我们发现点击span 打印span被点击,div被点击,body被点击 点击div 打印div被点击,body被点击

事件冒泡和事件捕获

  • 我们会发现默认情况下事件是从最内层的span向外依次传递的顺序,这个顺序我们称之为事件冒泡(Event Bubble)。
  • 事实上,还有另外一种监听事件流的方式就是从外层到内层(body -> span),这种称之为事件捕获(Event Capture);
  • 为什么会产生两种不同的处理流呢?
  • 这是因为早期浏览器开发时,不管是IE还是Netscape公司都发现了这个问题,但是他们采用了完全相反的事件流来对事件进行了传递;
  • IE采用了事件冒泡的方式,Netscape采用了事件捕获的方式;
  • 那么我们如何去监听事件捕获的过程呢?

addEventListener第三个参数默认为false,表示不捕获,可以手动传入true

document.body.addEventListener("click",()=>{
    console.log("body被点击")
},true)

冒泡和捕获的顺序

如果我们同时有事件冒泡和时间捕获的监听,那么会优先监听到事件捕获的:

代码演示


const spanEl = document.querySelector(".span")
const divEl = document.querySelector(".div")
spanEl.addEventListener("click", () => {
  console.log("事件冒泡:span元素被点击了")
})
divEl.addEventListener("click", () => {
  console.log("事件冒泡:div元素被点击了")
})
document.body.addEventListener("click", () => {
  console.log("事件冒泡:body元素被点击了")
})
// 再次监听
spanEl.addEventListener("click", (event) => {
  console.log("事件捕获:span元素被点击了")
  event.stopPropagation()
}, true)
divEl.addEventListener("click", () => {
  console.log("事件捕获:div元素被点击了")
}, true)
document.body.addEventListener("click", (event) => {
  console.log("事件捕获:body元素被点击了")
}, true)

点击spandom 输出应该是:

事件捕获:body元素被点击了

事件捕获:div元素被点击了

事件捕获:span元素被点击了

事件冒泡:span元素被点击了

事件冒泡:span元素被点击了

事件冒泡:body元素被点击了

事件对象event

  • 当一个事件发生时,就会有和这个事件相关的很多信息:
  • 比如事件的类型是什么,你点击的是哪一个元素,点击的位置是哪里等等相关的信息;
  • 那么这些信息会被封装到一个Event对象中;
  • 该对象给我们提供了想要的一些属性,以及可以通过该对象进行某些操作;
  • 常见的属性:
  • type:事件的类型;
  • target:当前事件发生的元素;
  • currentTarget:当前处理事件的元素;
  • offsetX、offsetY:点击元素的位置;
  • 常见的方法:
  • preventDefault:取消事件的默认行为;
  • stopPropagation:阻止事件的进一步传递;

代码演示


const spanEl = document.querySelector(".span")
spanEl.addEventListener("click", (event) => {
  console.log("span元素被点击:", event)
  console.log("事件的类型:", event.type)
  console.log("事件的元素:", event.target, event.currentTarget)
  console.log("事件发生的位置:", event.offsetX, event.offsetY)
})
const divEl = document.querySelector(".container")
divEl.addEventListener("click", (event) => {
  console.log("div元素被点击:", event.target, event.currentTarget)
})
// 常见的方法
// preventDefault
//取消事件的默认行为 ,比如a标签就是默认跳转
const aEl = document.querySelector("a")
aEl.addEventListener("click", (event) => {
  event.preventDefault()
})
// stopPropagation
// 组织事件进一步传递,其实就是取消冒泡
const aEl = document.querySelector("a")
aEl.addEventListener("click", (event) => {
  event.stopPropagation()
})

5f7960c7d6df9745728a755b4491963.png


相关文章
|
前端开发 Java 编译器
【前端学java】java基础练习缺少项目?看这篇文章就够了!(完结)
【8月更文挑战第11天】java基础练习缺少项目?看这篇文章就够了!(完结)
153 0
|
存储 前端开发 JavaScript
【前端学java】一篇文章带你精通java 中的数组(10)
【8月更文挑战第10天】一篇文章带你精通java 中的数组
148 3
【前端学java】一篇文章带你精通java 中的数组(10)
|
前端开发 JavaScript
这篇文章介绍了如何使用form表单结合Bootstrap格式将前端数据通过action属性提交到后端的servlet,包括前端表单的创建、数据的一级和二级验证,以及后端servlet的注解和参数获取。
这篇文章介绍了使用AJAX技术将前端页面中表单接收的多个参数快速便捷地传输到后端servlet的方法,并通过示例代码展示了前端JavaScript中的AJAX调用和后端servlet的接收处理。
这篇文章介绍了如何使用form表单结合Bootstrap格式将前端数据通过action属性提交到后端的servlet,包括前端表单的创建、数据的一级和二级验证,以及后端servlet的注解和参数获取。
|
存储 移动开发 前端开发
HTML5时代来临,这些新特性你掌握了吗?一篇文章带你玩转Web前端技术潮流!
【8月更文挑战第26天】HTML5(简称H5)作为新一代Web标准,相比HTML4带来了诸多增强功能。
509 2
|
前端开发 Java 编译器
【前端学java】java基础练习缺少项目?看这篇文章就够了!(17)
【8月更文挑战第11天】java基础练习缺少项目?看这篇文章就够了!
128 0
【前端学java】java基础练习缺少项目?看这篇文章就够了!(17)
|
前端开发 JavaScript 开发者
网页设计与开发:一篇文章带你领略前端世界的魅力
网页设计与开发:一篇文章带你领略前端世界的魅力
373 1
|
JavaScript 前端开发 网络架构
文本,展现到文章的面前,Vue结合v-for实现传参的方法,好的资料,Vue实现动态绑定,数据放到前端页面上
文本,展现到文章的面前,Vue结合v-for实现传参的方法,好的资料,Vue实现动态绑定,数据放到前端页面上
|
XML 存储 JavaScript
【前端】对前端小白极为友好的JS DOM入门文章
【前端】对前端小白极为友好的JS DOM入门文章
439 1
|
前端开发 JavaScript 网络安全
【网络安全XSS必知 | 前端开发基础】一篇文章速学 HTML
【网络安全XSS必知 | 前端开发基础】一篇文章速学 HTML
214 0
|
JSON 前端开发 JavaScript
【网络安全必备 | 前端开发基础】一篇文章速学 JavaScript
【网络安全必备 | 前端开发基础】一篇文章速学 JavaScript
234 0

热门文章

最新文章

  • 1
    前端如何存储数据:Cookie、LocalStorage 与 SessionStorage 全面解析
    886
  • 2
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(九):强势分析Animation动画各类参数;从播放时间、播放方式、播放次数、播放方向、播放状态等多个方面,完全了解CSS3 Animation
    403
  • 3
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(八):学习transition过渡属性;本文学习property模拟、duration过渡时间指定、delay时间延迟 等多个参数
    305
  • 4
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(七):学习ransform属性;本文学习 rotate旋转、scale缩放、skew扭曲、tanslate移动、matrix矩阵 多个参数
    278
  • 5
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(六):全方面分析css的Flex布局,从纵、横两个坐标开始进行居中、两端等元素分布模式;刨析元素间隔、排序模式等
    401
  • 6
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(五):背景属性;float浮动和position定位;详细分析相对、绝对、固定三种定位方式;使用浮动并清除浮动副作用
    590
  • 7
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(四):元素盒子模型;详细分析边框属性、盒子外边距
    635
  • 8
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(三):元素继承关系、层叠样式规则、字体属性、文本属性;针对字体和文本作样式修改
    193
  • 9
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(二):CSS伪类:UI伪类、结构化伪类;通过伪类获得子元素的第n个元素;创建一个伪元素展示在页面中;获得最后一个元素;处理聚焦元素的样式
    547
  • 10
    【CSS】前端三大件之一,如何学好?从基本用法开始吧!(一):CSS发展史;CSS样式表的引入;CSS选择器使用,附带案例介绍
    348