主頁 > 企業開發 > Javascript事件系統

Javascript事件系統

2020-10-06 06:40:31 企業開發

事件基礎

注意:本文不會深入探究Javascript的事件回圈,

提到事件,相信每位Javascript開發者都不會陌生,由于Javascript是先有實作,后有規范,因此,對于大部分人來說,事件模塊可以說是比較模糊的,本文將從不同角度幫助你理清楚事件模塊,

事件的本質可以說是一個回呼函式,當事件觸發時會呼叫你的監聽函式,

事件是一定會觸發的,如果沒有對應的監聽函式,就不會執行回呼,

比如下面就是用戶點擊指定元素列印日志的例子:

document.querySelector('#button').onclick = function() {    console.log('clicked');
};

事件基礎相信大家都沒什么問題,重點在后面的內容,

事件監聽方式

由于歷史原因,Javascript目前存在三種事件監聽方式:

  1. HTML代碼中監聽
  2. DOM0級監聽
  3. DOM2級監聽

Q: 為啥從DOM0級開始?

1998年,W3C綜合各瀏覽器廠商的現有API,指定了DOM1標準,在DOM1標準出現之前瀏覽器已有的事件監聽方式叫做DOM0級,

Q:DOM1級監聽到哪里去了?

由于DOM1標準只是對DOM0標準的整理+規范化,并沒有增加新的內容,因此DOM0級可以看做DOM1級,

HTML代碼監聽

<button onclick="alert('Hello World!')">點我</button>

直接將事件處理函式或事件處理代碼寫到HTML元素對應的屬性上的方式就是HTML代碼監聽方式

該方式有一個明顯的缺點,如果事件邏輯比較復雜時,將大段代碼直接寫在HTML元素上不利于維護,因此一般會提取到一個專一的函式進行處理,

<button onclick="callback()">點我</button>

該方式也有一個問題,那就是如果callback()函式還未加載好時點擊按鈕將報錯,而且直接將事件耦合到HTML元素上也不符合單一職責,HTML元素應該只負責展示,不負責事件,

不建議在開發中使用該方式處理事件,

DOM0級事件監聽

在DOM1級規范出來之前,各瀏覽器廠商已經提供了一套事件API,也就是DOM0級API,它的寫法如下:

<button id="click">點我</button><script>
  document.querySelector('#click').onclick = function() {    console.log('clicked');
  };</script>

這個相信大家在剛開始入行時寫的比較多,比如我們的ajax相關API就是DOM0級的,

var xhr = new XMLHttpRequest();
xhr.onload = function() {};
xhr.onerror = function() {};
DOM0級事件基本上都是以"on"開頭的

DOM0級事件也存在一個問題,那就是不支持添加多個事件處理函式,因此只有在不支持DOM2級事件的情況下才會使用DOM0級來系結事件,

DOM2級事件監聽

DOM2級事件是最新的事件處理程式規范(有許多年未更新了),DOM2級事件通過addEventListener方式給元素添加事件處理程式,

<button id="click">點我</button><script>
  document.addEventListener('click', function(){    console.log('clicked');
  });</script>

多次呼叫addEventListener可以系結多個事件處理程式,但是需要注意:

同樣的事件名、同樣的事件處理函式和同樣的事件流機制(冒泡和捕獲,下面會講到),只會觸發一次
// 下面的代碼只會觸發一次<button id="request">登錄</button><script>function onClick() {    console.log('clicked');
}document.querySelector('#request').addEventListener('click', onClick, false);document.querySelector('#request').addEventListener('click', onClick, false);</script>

onClick是同一個事件處理程式,所以只觸發一次

// 下面的代碼只會觸發兩次<button id="request">登錄</button>
<script>
document.querySelector('#request').addEventListener('click', function() {
    console.log('clicked');
}, false);
document.querySelector('#request').addEventListener('click', function() {
    console.log('clicked');
}, false);
</script>

兩個匿名函式,所以會觸發兩次

事件默認行為

很多網頁元素會有默認行為,比如下面這些:

  • 點擊a標簽的時候,會有跳轉行為
  • 點擊右鍵時會彈出選單
  • 在表單中點擊提交按鈕會提交表單

如果我們需要阻止默認行為,比如我們在阻止表單的默認提交事件,進行資料校驗,通過校驗后再呼叫表單submit方法提交,

不同的監聽方式阻止默認行為的方式也不同,

HTML代碼方式

HTML代碼方式支持return false和event.preventDefault()

return false方式

<form action="" onsubmit="return handleSubmit()">
    <button type="submit">Submit</button></form><script>function handleSubmit() {    return false;
}</script>

上例中我們監聽了表單的onsubmit事件,當點擊按鈕或者按下回車時,將會觸發handleSubmit方法,同時會阻止表單的提交,

表單內如果有type="submit"的按鈕存在,按下回車時就會自動提交,

HTML監聽方式阻止默認事件需要滿足以下兩點:

  1. HTML事件監聽代碼return handler()return不能少,少了就無法阻止默認行為
  2. handler()函式需要回傳false

event.preventDefault()

<a href="https://www.ddhigh.com" onclick="handleClick(event)" id="click">Href</a><script>function handleClick(e) {
    e.preventDefault();
}</script>

DOM0級事件方式

DOM0級事件支持return false和event.preventDefault()兩種方式,

event.preventDefault()

// event.preventDefault()<a href="https://www.ddhigh.com" id="click">Href</a><script>
    document.querySelector('#click').onclick= function (event) {
      event.preventDefault();
  };</script>

return false

// return false<a href="https://www.ddhigh.com" id="click">Href</a><script>
    document.querySelector('#click').onclick= function (event) {      return false;
  };</script>

兩種方式都能作業,不過建議使用event.preventDefault(),原因在下面DOM2級會講到

DOM2級事件

DOM2級事件事件只支持event.preventDefault()方式,這也是事件的標準處理方法,
<a href="https://www.ddhigh.com" id="click">Href</a><script>document.querySelector('#click').addEventListener('click', function (e) {
    e.preventDefault();
});</script>

事件冒泡與事件捕獲

先來看一個HTML結構

<div id="father">
  <div id="child">
    <div id="son">Click</div>
  </div></div>

我們知道,一旦系結了事件處理程式,在事件觸發時,事件處理函式都會觸發,

如果我們給father/child/son都系結了事件處理函式,點擊了son時,誰被觸發呢?

事實上,三個函式都會被觸發,因為son時child的子元素,child又是father的子元素,點擊son,同時也點擊了father和child,

由此帶來一個問題,三個函式誰先觸發,誰后觸發呢?這就是我們常說的事件流,father->child->son這種路徑是可以的,但是son->child->father這種路徑也是可以的,

針對這兩種方式,W3C給了我們一個答案,兩種方式都支持,即可以從父元素到子元素,又可以從子元素到父元素,前者叫事件捕獲,后者叫事件冒泡

事件捕獲

事件發生時采取自上而下的方式進行觸發,最先觸發的是window,其次是document,然后根據DOM層級依次觸發,最終進入到真正的事件元素,

addEventListener第三個引數傳入true就是捕獲方式的標志,
<div id="father">
    <div id="child">
      <div id="son">Click</div>
    </div>
    </div>

  <script>
  document.querySelector('#father').addEventListener('click', function () {      console.log('father');
  }, true);  document.querySelector('#child').addEventListener('click', function () {      console.log('child');
  }, true);  document.querySelector('#son').addEventListener('click', function () {      console.log('son');
  }, true);    </script>

點擊son之后的輸出順序為

father
child
son

事件冒泡

事件發生時采取自下而上的方式進行觸發,最先觸發的是發生事件的元素,其次是父元素,依次向上,最終觸發到documentwindow

addEventListener第三個引數傳入false就是事件冒泡的標志,
<div id="father">
    <div id="child">
      <div id="son">Click</div>
    </div>
    </div>

  <script>
  document.querySelector('#father').addEventListener('click', function () {      console.log('father');
  }, false);  document.querySelector('#child').addEventListener('click', function () {      console.log('child');
  }, false);  document.querySelector('#son').addEventListener('click', function () {      console.log('son');
  }, false);    </script>

點擊son之后的輸出順序為

son
child
father
由于事件捕獲和事件冒泡機制,我們需要一個標記來標識真正觸發事件的元素,這個元素就是event.target,而另外一個相似的屬性叫event.currentTarget,這是當前元素,

事件捕獲和時間冒泡的順序

根據瀏覽器規范,事件捕獲會先于事件冒泡發生,因此,總的事件順序如下
  1. window 捕獲階段
  2. document 捕獲階段
  3. ... 依次到真正觸發事件的元素 捕獲階段
  4. 真正觸發事件的元素 冒泡階段
  5. 依次向上的父元素 冒泡階段
  6. document 冒泡階段
  7. window 冒泡階段
<div id="father">
        <div id="child">
            <div id="son">Click</div>
        </div>
    </div>

    <script>
        document.querySelector('#father').addEventListener('click', function () {            console.log('father捕獲');
        }, true);        document.querySelector('#child').addEventListener('click', function () {            console.log('child捕獲');
        }, true);        document.querySelector('#son').addEventListener('click', function () {            console.log('son捕獲');
        }, true);        document.querySelector('#father').addEventListener('click', function () {            console.log('father冒泡');
        }, false);        document.querySelector('#child').addEventListener('click', function () {            console.log('child冒泡');
        }, false);        document.querySelector('#son').addEventListener('click', function () {            console.log('son冒泡');
        }, false);    </script>

點擊son之后的輸出為

father捕獲
child捕獲
son捕獲
son冒泡
child冒泡
father冒泡

事件系結和事件委托

弄明白瀏覽器的事件流機制之后,來討論事件系結和事件委托其實是很簡單的事情,

事件系結

就是在事件監聽方式中直接對具體元素進行事件監聽的方式,有個明顯的缺點,對于新增加的DOM節點是無法監聽到事件的,

    <div >click1</div>
    <div >click2</div>
    <script>
        document.querySelectorAll('.a').forEach(ele => ele.onclick = function () {            console.log('clicked ' + this.innerHTML);
        });
        setTimeout(function () {            const div3 = document.createElement('div')
            div3.className = "a";
            div3.innerHTML = "click3"
            document.body.appendChild(div3)
        }, 500);    </script>

上面的click3點擊是沒有任何反應的,因為在創建該元素時沒有系結事件處理函式,

事件委托

我們利用事件流機制來實作上面的需求,

事件委托就是利用事件流機制,在父元素進行監聽,由于事件冒泡機制,父元素可以接受新添加元素的事件,
    <div >click1</div>
    <div >click2</div>
    <script>
        document.body.addEventListener('click', function (e) {            console.log(e.target.innerHTML)
        }, false);
        setTimeout(function () {            const div3 = document.createElement('div')
            div3.className = "a";
            div3.innerHTML = "click3"
            document.body.appendChild(div3)
        }, 500);    </script>

由于事件冒泡機制,click3元素點擊之后會將事件冒泡給父元素,也就是我們的document.body,通過event.target可以拿到真正觸發事件的元素,

轉載請註明出處,本文鏈接:https://www.uj5u.com/qiye/158919.html

標籤:JavaScript

上一篇:JS高級---為內置物件添加原型方法

下一篇:JS-DOM事件

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • IEEE1588PTP在數字化變電站時鐘同步方面的應用

    IEEE1588ptp在數字化變電站時鐘同步方面的應用 京準電子科技官微——ahjzsz 一、電力系統時間同步基本概況 隨著對IEC 61850標準研究的不斷深入,國內外學者提出基于IEC61850通信標準體系建設數字化變電站的發展思路。數字化變電站與常規變電站的顯著區別在于程序層傳統的電流/電壓互 ......

    uj5u.com 2020-09-10 03:51:52 more
  • HTTP request smuggling CL.TE

    CL.TE 簡介 前端通過Content-Length處理請求,通過反向代理或者負載均衡將請求轉發到后端,后端Transfer-Encoding優先級較高,以TE處理請求造成安全問題。 檢測 發送如下資料包 POST / HTTP/1.1 Host: ac391f7e1e9af821806e890 ......

    uj5u.com 2020-09-10 03:52:11 more
  • 網路滲透資料大全單——漏洞庫篇

    網路滲透資料大全單——漏洞庫篇漏洞庫 NVD ——美國國家漏洞庫 →http://nvd.nist.gov/。 CERT ——美國國家應急回應中心 →https://www.us-cert.gov/ OSVDB ——開源漏洞庫 →http://osvdb.org Bugtraq ——賽門鐵克 →ht ......

    uj5u.com 2020-09-10 03:52:15 more
  • 京準講述NTP時鐘服務器應用及原理

    京準講述NTP時鐘服務器應用及原理京準講述NTP時鐘服務器應用及原理 安徽京準電子科技官微——ahjzsz 北斗授時原理 授時是指接識訓通過某種方式獲得本地時間與北斗標準時間的鐘差,然后調整本地時鐘使時差控制在一定的精度范圍內。 衛星導航系統通常由三部分組成:導航授時衛星、地面檢測校正維護系統和用戶 ......

    uj5u.com 2020-09-10 03:52:25 more
  • 利用北斗衛星系統設計NTP網路時間服務器

    利用北斗衛星系統設計NTP網路時間服務器 利用北斗衛星系統設計NTP網路時間服務器 安徽京準電子科技官微——ahjzsz 概述 NTP網路時間服務器是一款支持NTP和SNTP網路時間同步協議,高精度、大容量、高品質的高科技時鐘產品。 NTP網路時間服務器設備采用冗余架構設計,高精度時鐘直接來源于北斗 ......

    uj5u.com 2020-09-10 03:52:35 more
  • 詳細解讀電力系統各種對時方式

    詳細解讀電力系統各種對時方式 詳細解讀電力系統各種對時方式 安徽京準電子科技官微——ahjzsz,更多資料請添加VX 衛星同步時鐘是我京準公司開發研制的應用衛星授時時技術的標準時間顯示和發送的裝置,該裝置以M國全球定位系統(GLOBAL POSITIONING SYSTEM,縮寫為GPS)或者我國北 ......

    uj5u.com 2020-09-10 03:52:45 more
  • 如何保證外包團隊接入企業內網安全

    不管企業規模的大小,只要企業想省錢,那么企業的某些服務就一定會采用外包的形式,然而看似美好又經濟的策略,其實也有不好的一面。下面我通過安全的角度來聊聊使用外包團的安全隱患問題。 先看看什么服務會使用外包的,最常見的就是話務/客服這種需要大量重復性、無技術性的服務,或者是一些銷售外包、特殊的職能外包等 ......

    uj5u.com 2020-09-10 03:52:57 more
  • PHP漏洞之【整型數字型SQL注入】

    0x01 什么是SQL注入 SQL是一種注入攻擊,通過前端帶入后端資料庫進行惡意的SQL陳述句查詢。 0x02 SQL整型注入原理 SQL注入一般發生在動態網站URL地址里,當然也會發生在其它地發,如登錄框等等也會存在注入,只要是和資料庫打交道的地方都有可能存在。 如這里http://192.168. ......

    uj5u.com 2020-09-10 03:55:40 more
  • [GXYCTF2019]禁止套娃

    git泄露獲取原始碼 使用GET傳參,引數為exp 經過三層過濾執行 第一層過濾偽協議,第二層過濾帶引數的函式,第三層過濾一些函式 preg_replace('/[a-z,_]+\((?R)?\)/', NULL, $_GET['exp'] (?R)參考當前正則運算式,相當于匹配函式里的引數 因此傳遞 ......

    uj5u.com 2020-09-10 03:56:07 more
  • 等保2.0實施流程

    流程 結論 ......

    uj5u.com 2020-09-10 03:56:16 more
最新发布
  • 使用Django Rest framework搭建Blog

    在前面的Blog例子中我們使用的是GraphQL, 雖然GraphQL的使用處于上升趨勢,但是Rest API還是使用的更廣泛一些. 所以還是決定回到傳統的rest api framework上來, Django rest framework的官網上給了一個很好用的QuickStart, 我參考Qu ......

    uj5u.com 2023-04-20 08:17:54 more
  • 記錄-new Date() 我忍你很久了!

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 大家平時在開發的時候有沒被new Date()折磨過?就是它的諸多怪異的設定讓你每每用的時候,都可能不小心踩坑。造成程式意外出錯,卻一下子找不到問題出處,那叫一個煩透了…… 下面,我就列舉它的“四宗罪”及應用思考 可惡的四宗罪 1. Sa ......

    uj5u.com 2023-04-20 08:17:47 more
  • 使用Vue.js實作文字跑馬燈效果

    實作文字跑馬燈效果,首先用到 substring()截取 和 setInterval計時器 clearInterval()清除計時器 效果如下: 實作代碼如下: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta ......

    uj5u.com 2023-04-20 08:12:31 more
  • JavaScript 運算子

    JavaScript 運算子/運算子 在 JavaScript 中,有一些運算子可以使代碼更簡潔、易讀和高效。以下是一些常見的運算子: 1、可選鏈運算子(optional chaining operator) ?.是可選鏈運算子(optional chaining operator)。?. 可選鏈操 ......

    uj5u.com 2023-04-20 08:02:25 more
  • CSS—相對單位rem

    一、概述 rem是一個相對長度單位,它的單位長度取決于根標簽html的字體尺寸。rem即root em的意思,中文翻譯為根em。瀏覽器的文本尺寸一般默認為16px,即默認情況下: 1rem = 16px rem布局原理:根據CSS媒體查詢功能,更改根標簽的字體尺寸,實作rem單位隨螢屏尺寸的變化,如 ......

    uj5u.com 2023-04-20 08:02:21 more
  • 我的第一個NPM包:panghu-planebattle-esm(胖虎飛機大戰)使用說明

    好家伙,我的包終于開發完啦 歡迎使用胖虎的飛機大戰包!! 為你的主頁添加色彩 這是一個有趣的網頁小游戲包,使用canvas和js開發 使用ES6模塊化開發 效果圖如下: (覺得圖片太sb的可以自己改) 代碼已開源!! Git: https://gitee.com/tang-and-han-dynas ......

    uj5u.com 2023-04-20 08:01:50 more
  • 如何在 vue3 中使用 jsx/tsx?

    我們都知道,通常情況下我們使用 vue 大多都是用的 SFC(Signle File Component)單檔案組件模式,即一個組件就是一個檔案,但其實 Vue 也是支持使用 JSX 來撰寫組件的。這里不討論 SFC 和 JSX 的好壞,這個仁者見仁智者見智。本篇文章旨在帶領大家快速了解和使用 Vu ......

    uj5u.com 2023-04-20 08:01:37 more
  • 【Vue2.x原始碼系列06】計算屬性computed原理

    本章目標:計算屬性是如何實作的?計算屬性快取原理以及洋蔥模型的應用?在初始化Vue實體時,我們會給每個計算屬性都創建一個對應watcher,我們稱之為計算屬性watcher ......

    uj5u.com 2023-04-20 08:01:31 more
  • http1.1與http2.0

    一、http是什么 通俗來講,http就是計算機通過網路進行通信的規則,是一個基于請求與回應,無狀態的,應用層協議。常用于TCP/IP協議傳輸資料。目前任何終端之間任何一種通信方式都必須按Http協議進行,否則無法連接。tcp(三次握手,四次揮手)。 請求與回應:客戶端請求、服務端回應資料。 無狀態 ......

    uj5u.com 2023-04-20 08:01:10 more
  • http1.1與http2.0

    一、http是什么 通俗來講,http就是計算機通過網路進行通信的規則,是一個基于請求與回應,無狀態的,應用層協議。常用于TCP/IP協議傳輸資料。目前任何終端之間任何一種通信方式都必須按Http協議進行,否則無法連接。tcp(三次握手,四次揮手)。 請求與回應:客戶端請求、服務端回應資料。 無狀態 ......

    uj5u.com 2023-04-20 08:00:32 more