程序员社区

js事件模型详解

javascript事件模型其实就是观察者模式的体现,就是当事件触发时,监听该事件的全部触发事件都会被调用。

一、事件流

事件流主要分为两种,一种是事件捕获,另一种是事件冒泡。
事件捕获:事件从上向下进行触发,事件首先从最外侧父元素开始触发,然后一直向里延伸,知道触发到最内层的元素。
事件冒泡:事件从下向上进行触发,事件首先会触发最底层的子节点,然后一直向上触发父元素事件。

二、事件

1、DOM0事件,在不同的浏览器上面不存在兼容性问题,不过在触发事件中,默认使用的是事件冒泡。

//直接在html元素中绑定
<div id = "box1" onclick="func1()"></div>

//使用js获取元素标签绑定函数
let box1 = document.getElementbyId("box1")
box1.onclick = function() {
	XXXXX
}

//移除监听函数
box1.onclick = null
IE事件模型
IE事件模型主要分为两个阶段:一个是事件处理阶段,就是当事件到达该元素时,执行相关函数阶段。
第二个是事件冒泡阶段:执行完事件阶段一,然后就会执行进行事件冒泡。

//ie事件监听
//监听事件
attachEvent(eventType, handler)
//移除事件
detachEvent(eventType, handler)
//参数解析
eventType:事件类型,需要加上on
handler:事件处理函数。
//代码解析
<body>

  <div id="box1">
    <div id="box2"></div>
  </div>

  <button id="btn">点击移除box1事件</button>


  <script>

    var box1 = document.getElementById('box1');
    var box2 = document.getElementById("box2");
    var btn = document.getElementById("btn");
    function func1() {
      console.log("执行box1")
    }
    function func2() {
      console.log("执行box2")
    }
    box1.attachEvent("onclick", func1)

    box2.attachEvent("onclick", func2)

    btn.attachEvent("onclick", function () {
      box1.detachEvent("onclick", func1)
    })
  </script>
</body>
DOM2级模型
DOM2级模型的事件,除了IE6-IE8不支持外,其他的都支持。
该事件中主要存在三个过程:
事件捕获:从最外层父元素从向最内层子元素中进行事件传递,如果存在事件处理函数,则执行。
事件处理:执行最内层的时间。
事件冒泡:从最内层子元素向最外层元素进行事件传递,如果存在事件处理函数,则执行。

//事件处理函数
//监听事件
addEventListener(eventType, handler, useCapture)
//移除事件
removeEventListener(eventType, handler, useCapture)
//代码展示
  <script>

    var box1 = document.getElementById('box1');
    var box2 = document.getElementById("box2");
    box1.addEventListener("click", function() {
      console.log("box1")
    },false)
    box2.addEventListener("click",function() {
      console.log("box2")
    },false)
  </script>
第三个参数来设置时事件捕获还是事件冒泡。默认是事件冒泡(false.

三、事件对象

当事件触发时,默认会创建一个新的对象,并且该对象携带一些属性和方法。并且该对象会作为第一个参数传递给监听函数。
DOM事件中常见的属性:
	type:用于获取事件类型
	target:获取事件目标
	stopPropagation()阻止事件冒泡
	preventDefault()阻止事件默认行为
IE事件模型常见的属性:
	type用于获取事件类型
	srcElement获取事件目标
	cancelBubble阻止事件冒泡
	returnValue阻止事件默认行为

四、event wrapper

因为存在DOM0和DOM2和IE事件模型,所以我们可以对其进行统一封装来统一调用。这样来解决兼容性问题。
const EventUtils = {
  // 主要从DOM0|DOM2|IE事件模型来封装
  // 监听事件
  addEvent: function(element, type, handler) {
    if(element.addEventListener) {
      element.addEventListener(type, handler, false)
    }else if(element.attachEvent) {
      element.attachEvent("on" + type, handler)
    }else {
      element["on" + type] = handler
    }
  },
  // 移除事件
  removeEvent: function(element, type, handler) {
    if(element.removeEventListener) {
      element.removeEventListener(type, handler, false)
    }else if(element.detachEvent) {
      element.detachEvent("on" + type, handler)
    }else {
      element["on" + type] = handler
    }
  },
  // 获取事件目标
  getTarget: function(event) {
    return event.target || event.srcElement;
  },
  // 获取event对象的引用,取到事件的所有信息,确保随时使用event
  getEvent: function(event) {
    return event || window.event
  },
  // 组织默认事件
  stopPropagation: function(event) {
    if(event.stopPropagation) {
      event.stopPropagation()
    }else{
      event.cancelBubble = true
    }
  },
  // 取消事件的默认行为
  preventDefault: function(event) {
    if(event.preventDefault) {
      event.preventDefault()
    }else {
      event.returnValue = false
    }
  }

}

五、事件代理/事件委托

事件代理,也称为事件委托,就是将子元素中的事件全部放到父元素中进行触发。
<body>

  <div id="box">
    <button id="btn1">按钮1</button>
    <button id="btn2">按钮2</button>
    <button id="btn3">按钮3</button>
  </div>

  <script>
    let box = document.getElementById('box')
    let btn1 = document.getElementById("btn1")
    let btn2 = document.getElementById("btn2")
    let btn3 = document.getElementById("btn3")

    box.addEventListener("click", function(event) {
      if(event.target.getAttribute("id") === "btn1") {
        console.log("您点击的是按钮一")
      }else if(event.target.getAttribute("id") === "btn2") {
        console.log("您点击的是按钮二")
      }else {
        console.log("您点击的是按钮三")
      }
    })
  </script>

</body>
这样让所有事件全部由父元素进行处理,利用的是事件冒泡。
在react中的事件使用的也是事件委托。

赞(0) 打赏
未经允许不得转载:IDEA激活码 » js事件模型详解

一个分享Java & Python知识的社区