JavaScript事件

2021-03-26  本文已影响0人  Joemoninni

1、事件流:

事件流描述了页面接收事件的顺序。IE支持事件冒泡流,而网景支持事件捕获流。

1-1 事件冒泡:

IE事件流被称为事件冒泡。事件被定义为从最具体的元素开始触发,向上传播至没那么具体的元素(文档),即从内向外

1-2 事件捕获:

网景提出另外一种名为事件捕获的事件流。意思是最不具体的节点应该先收到事件,而最具体的节点最后收到事件,即从外向内

1-3 DOM事件流:

DOM事件流分为3个阶段:事件捕获、到达目标和事件冒泡。事件捕获最先发生,为提前拦截事件提供了可能。然后就是目标阶段,实际的目标元素接收到事件。最后一个阶段是冒泡,最迟要在这个阶段响应事件。

2、事件处理程序:

事件意味着用户或浏览器执行的某种动作,如cilckloadmouseover。为响应事件而调用的函数被称为事件处理程序(或事件监听器),事件处理程序的名字以"on"开头,因此click事件的处理程序叫onclick,同理,有onload等等

2-1 HTML事件处理程序

<button onclick="console.log('clicked')">按钮</button>
<!-- 或者 -->
<button onclick="showMessage()">按钮</button>
<script>
    function showMessage() {
        console.log('clicked')
    }
</script>

2-2 DOM0事件处理程序

<button id="button"></button>
<script>
    let btn = document.getElementById('button')
    // 添加事件处理程序
    btn.onclick = function() {
        console.log(this.id) // button
    }
    // 移除事件处理程序
    btn.onclick = null
</script>

要使用这种方式,必须先取得要操作对象的引用。像这样为事件处理程序赋值时,所赋函数被视为元素的方法,因此,事件处理程序会在元素的作用域中运行,即this等于元素。在事件处理程序里通过this可以访问元素的任何属性和方法。以这种方式添加事件处理程序是注册在事件流的冒泡阶段的。

2-3 DOM2事件处理程序

事件处理程序赋值addEventListener()

事件处理程序移除removeEventListener()

这两个方法暴露在所有DOM节点上,它们接收3个参数:事件名、事件处理函数和一个布尔值。true表示在捕获阶段调用事件处理函数,false(默认值)表示在冒泡阶段调用事件处理函数。

let btn = document.getElementById('button')
btn.addEventListener('click', () => {
    cocnsole.log(this.id); // button this指向元素本身
}, false)

优势:可以为同一个事件添加多个事件处理程序。多个事件处理程序以添加的顺序来触发。

注意点:

2-4 IE事件处理程序

事件处理程序赋值attachEvent()

事件处理程序移除detachEvent()

这两个方法接收2个参数:事件处理程序的名字on + 'type'和事件处理函数。只会添加到冒泡阶段。

let btn = document.getElementById('button')
btn.attachEvent('onclick', function() {
    console.log(this) // window 由于事件处理程序是在全局作用域下运行的,所以this指向window
})

优势:也是可以添加多个事件处理程序。不过这里多个事件处理程序以添加的顺序反向触发。

注意点:通过attachEvent()添加事件处理程序只能使用detachEvent()来移除,需提供相同的参数。不支持匿名函数。

2-5 跨浏览器事件处理程序❗❗❗

<button id="btn" onclick="showMessage()">click me</button>
<script>
    let EventUtil = {
      addHandler: function(ele, type, handler) {
        if (ele.addEventListener) {
          ele.addEventListener(type, handler, false);
        } else if (ele.attachEvent) {
          ele.attachEvent('on' + type, handler);
        } else {
          ele['on' + type] = handler;
        }
      },
      removeHandler: function(ele, type, handler) {
        if (ele.removeEventListener) {
          ele.removeEventListener(type, handler, false);
        } else if (ele.detachEvent) {
          ele.detachEvent('on' + type, handler);
        } else {
          ele['on' + type] = null;
        }
      }
    }
    let btn = document.getElementById('button');
    let handler = function() {
      console.log('clicked')
    }
    EventUtil.addHandler(btn, 'click', handler)
    EventUtil.removeHandler(btn, 'click', handler)
</script>

3、事件对象

DOM中发生事件时,所有相关信息都会被收集并存储在一个名为event的对象中。这个对象包含一些基本信息,如导致事件的元素、发生的事件类型。

3-1 DOM事件对象(含DOM0DOM2)

DOM合规的浏览器中,event对象是传给事件处理程序的唯一参数。不管以哪种方式(DOM0/DOM2)指定的事件处理程序,都会传入这个event对象。

let btn = document.getElementById('button');
btn.onclick = function(e) {
    console.log(e.type); // click
}
btn.addEventListener("click", (e) => {
    console.log(e.type); // click
}, false)

需要掌握的事件对象的方法:

preventDefault():用于阻止特定事件的默认动作。如取消a标签在被点击时导航到href属性指定的URL这个默认行为。

<a id="myLink" href="###">链接</a>
<script>
    let link = document.getElementById('myLink');
    link.onclick = function(e) {
        e.preventDefault();
    }
</script>

stopPropagation():用于立即阻止事件流在DOM结构中传播,取消后续的事件捕获或冒泡(既可取消捕获,也可取消冒泡)。如直接在添加到按钮的事件处理程序中调用stopPropagation(),可以阻止document.body上注册的事件处理程序执行。

let btn = document.getElementById('button');
btn.onclick = function(e) {
    console.log('clicked')
    e.stopPropagation()
}
document.body.onclick = function(e) {
    console.log('body clicked')
}

eventPhase:用于确定事件流当前所处的阶段。1是捕获阶段 2是目标阶段 3是冒泡阶段

target:事件目标元素

3-2 IE事件对象

DOM事件对象不同,IE事件对象可以基于事件处理程序被指定的方式以不同的方式来访问:

如果事件处理程序是使用DOM0方式来指定的,则event对象是window的一个属性

let btn = document.getElementById('button');
btn.onclick = function() {
    let event = window.event;
    console.log(event.type)
}

如果事件处理程序是使用IE事件处理程序,即attachEvent()方式来指定的,则event对象是作为参数传给处理函数

let btn = document.getElementById('button');
btn.attachEvent("click", (event) => {
    console.log(event.type)
})

如果事件处理程序是使用HTML属性方式来指定的,则event对象可以通过变量event来访问

<button onclick="console.log(event.type)">click</button>

需要掌握的事件对象的属性或方法:

returnValue:用于取消给定事件默认的行为。

btn.onclick = function() {
    window.event.returnValue = false; // 此属性设置为false, 为阻止默认动作
}

cancelBubble:用于阻止事件冒泡。(由于IE8及更早的版本不支持捕获阶段,所以只会取消冒泡)

btn.onclick = function() {
    console.log('clicked')
    window.event.cancelBubble = true; // 此属性设置为true, 为阻止事件冒泡
}
document.body.onclick = function() {
    console.log('body clicked')
}

srcElement:事件目标元素

3-3 跨浏览器事件对象❗❗❗

<button id="btn">click me</button>
<script>
    var EventUtil = {
        addHandler: ...,
        removeHandler: ...,
        // 获取事件对象
        getEvent: function(event) {
            return event ? event : window.event
        },
        // 获取事件目标
        getTarget: function(event) {
            return event.target || event.srcElement
        },
        // 阻止默认事件
        preventDefault: function(event) {
            if (event.preventDefault) {
                event.preventDefault();
            } else {
                event.returnValue = false;
            }
        },
        // 阻止事件冒泡或事件捕获
        stopPropagation: function(event) {
            if (event.stopPropagation) {
                event.stopPropagation()
            } else {
                event.cancelBubble = true;
            }
        }
    }
</script>

4、事件类型

这里不一一列举。具体查看红宝书17章第四节

5、内存与性能

JavaScript中,页面中事件处理程序的数量与页面整体性能直接相关。原因有很多,首先,每个函数都是对象,都占用内存空间,对象越多,性能越差。其次,为执行事件处理程序所需访问DOM的次数会造成整个页面交互的延迟。

5-1 事件委托(事件冒泡的应用)

”过多事件处理程序“的解决方案是使用事件委托。事件委托利用事件冒泡,可以只使用一个事件处理程序来管理一个类型的事件。比如有一个li列表,需要给每一项添加click事件,如果量少的话可能没有感觉什么变化,但是如果量大的话,上万个,几十万个,那挨个添加事件处理程序显然不是很🆗的做法。解决办法是利用事件冒泡,使用事件委托,只要给所有元素共同的祖先节点添加一个事件处理程序即可解决。

事件委托的优点:

  1. 只要页面渲染出可点击的元素,就可以无延迟地起作用
  2. 设置页面事件处理程序时省时省力,节省DOM的引用
  3. 减少整个页面所需的内存,提升整体性能

5-2 删除事件处理程序

把事件处理程序指定给元素后,在浏览器代码和负责页面交互的JavaScript代码之间就建立了联系。这种联系建立得越多,页面性能越差。除了上面说的使用事件委托来限制这种连接之外,还应该及时删除不再使用的事件处理程序。

导致这个问题的原因主要有两个:

  1. 删除带有事件处理程序的元素时:因为该元素被删除后,其实它仍然关联着一个事件处理程序,很有可能元素的引用和事件处理程序的引用都会残留在内存中。所以为了保险起见,我们如果知道某个元素会被删除,最后在删除它之前手动移除它的事件处理程序。
  2. 页面卸载时:同样的道理,页面卸载时,其实页面上的事件处理程序仍然被引用着,没有清理,则残留在内存中。之后,浏览器每次加载和卸载页面,内存中残留对象的数量都会增加,这是因为事件处理程序不会被回收。所以,举个比较常用的例子,一般我们在vue页面中使用事件监听,都要在页面卸载之前也就是beforeDestroy生命周期中移除事件监听。
上一篇下一篇

猜你喜欢

热点阅读