JavaScript複習——04 事件

来源:https://www.cnblogs.com/trueasureyuki/archive/2023/11/12/17822779.html
-Advertisement-
Play Games

事件對象 事件對象是由瀏覽器在外面觸發事件的時候創建的,這個對象封裝了各種事件相關的各種信息 例如: 滑鼠的位置 鍵盤的按鍵 瀏覽器創建事件對象後,會將事件對象作為響應參數傳遞 在DOM類型中有多種不同類型的事件對象,但是他們都一個祖先Event event.clientX:獲取滑鼠的X軸坐標 ev ...


事件對象

事件對象是由瀏覽器在外面觸發事件的時候創建的,這個對象封裝了各種事件相關的各種信息

例如:

  1. 滑鼠的位置
  2. 鍵盤的按鍵

瀏覽器創建事件對象後,會將事件對象作為響應參數傳遞
在DOM類型中有多種不同類型的事件對象,但是他們都一個祖先Event

  1. event.clientX:獲取滑鼠的X軸坐標
  2. event.clientY:獲取滑鼠的Y軸坐標
  3. event.target:表示觸發事件的對象
  4. event.currentTarget:綁定事件的對象
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>DOM</title>
      <style>
          .box1 {
              width:300px;
              height:300px;
              background-color: #bfa;
          }
      </style>
  </head>
  <body>
      <div id="box1" class="box1">
          <div id="box2">
              
          </div>
      </div>
    <script>
        const box1 = document.getElementById("box1")
        const box2 = document.getElementById("box2")
        box1.onclick = function(event) {
			console.log(event.target)
            console.log(this)
        } 
        box2.onclick = function(event) {
			alert('我是box2')
        } 
    </script>
  </body>
</html>

事件冒泡

事件冒泡:當我們的元素上某個事件被觸發(內部元素不一定要綁定事件)後,其祖先元素的相同事件也會被觸發,冒泡大大的簡化了我們代碼的編寫,但是在某些場景下,我們有時也不希望冒泡

  1. event.stopPropagation():取消事件傳導
  2. event.preventDefault():取消事件的預設行為
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>DOM</title>
      <style>
          .box1 {
              width:300px;
              height:300px;
              background-color: #bfa;
          }
          #box {
              width:100px;
              height:100px;
              background-color: orange;
          }
      </style>
  </head>
  <body>
      <div id="box1" class="box1">
          <div id="box2">
              
          </div>
      </div>
    <script>
        const box1 = document.getElementById("box1")
        const box2 = document.getElementById("box2")
        box1.onclick = function(event) {
			console.log(event.target)
            console.log(this)
        } 
        box2.onclick = function(event) {
            // 停止box2點擊事件的傳導
            event.stopPropagation()
			alert('我是box2')
        } 
    </script>
  </body>
</html>

冒泡為什麼有利

當我們需要所有元素都有一個共有的事件時,我們如果沒有冒泡就需要一個個給他們添加事件
如果我們有冒泡,就只需要給它們的共同祖先元素添加一個事件就可以,這就是冒泡的好處
因為我們自己元素點擊之後,事件會向上冒泡,所以父級元素的處理器也會感知到


事件的委托

我們希望,只綁定一次事件,就可以讓所有的超鏈接和未來新添加的都具有這些事件

思路:我們可以將事件統一綁定document,這樣點擊超鏈接之後,由於事件的冒泡會導致document的事件被觸發,這樣就只綁定了一次,所有的超鏈接都會具有這些事件
但是這也帶來了一個問題,會讓除了超鏈接外的元素也觸發事件

委托:就是本該綁定給多個對象的事件,綁定給它們的共同祖先元素,這樣就可以降低我們代碼的複雜度

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>DOM</title>
  </head>
  <body>
      <button type="button" id="btn">
          點我一下
      </button>
      <ul id="list">
          <li><a href="javascript:;">鏈接1</a></li>
          <li><a href="javascript:;">鏈接2</a></li>
          <li><a href="javascript:;">鏈接3</a></li>
          <li><a href="javascript:;">鏈接4</a></li>
      </ul>
    <script>
        const links = document.links;
        const list = document.getElementById('list')
        const btn = document.getElementById('btn')
        for(let i = 0; i < links.length; i++) {
            links[i].addEventListener('click',(event)=>{
                alert(event.target.textContent)
            })
        }
        // 點擊按鈕之後新添加一個li
        /*
        	我們新添加的鏈接並不會綁定事件
        */
        btn.addEventListener('click',()=>{
            list.insertAdjancentHTML("beforeend","<li><a href="javascript:;">新鏈接</a><</li>")
        })
    </script>
  </body>
</html>

處理委托帶來的問題

我們之前也說了問題的帶來的問題,我們在給共有父元素綁定事件的時候,只要是子元素都有可能觸發事件,那麼我們如果要選擇對應的子元素觸發事件呢?

可以用以下的方式來解決

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>DOM</title>
  </head>
  <body>
      <button type="button" id="btn">
          點我一下
      </button>
      <ul id="list">
          <li><a href="javascript:;">鏈接1</a></li>
          <li><a href="javascript:;">鏈接2</a></li>
          <li><a href="javascript:;">鏈接3</a></li>
          <li><a href="javascript:;">鏈接4</a></li>
      </ul>
    <script>
      
        const list = document.getElementById('list')
        const links = list.getElementByTagName('li');
        const btn = document.getElementById('btn')
       
        document.addEventListener("click",(event)=>{
            if([...links].includes(event.target)){
				alert('超鏈接事件對象觸發了')
            } 
        })
        // 點擊按鈕之後新添加一個li
        /*
        	我們新添加的鏈接並不會綁定事件
        */
        btn.addEventListener('click',()=>{
            list.insertAdjancentHTML("beforeend","<li><a href="javascript:;">新鏈接</a><</li>")
        })
    </script>
  </body>
</html>

這上面的代碼必須是li才能觸發事件,要不然是不能觸發事件的


事件的捕獲

在DOM中,事件的傳播分為三個階段

  1. 捕獲階段
    • 事件的捕獲指的是我們的事件從外向內傳導
    • 由祖先元素向目標元素進行事件的捕獲
  2. 目標階段(觸發事件的對象)
  3. 冒泡階段
    • 由目標元素想祖先元素進行冒泡從內向外傳導

如果我們希望在捕獲階段觸發事件,那麼我們就需要將addEventListener()的第三個參數設置為true

註意:我們一般不希望在捕獲階段觸發事件,設置了捕獲事件,冒泡就會失效

  1. evnet.eventPhase:事件觸發的時機
    1. 0:表示沒有觸發事件
    2. 1:是捕獲階段
    3. 2:表示為目標階段
    4. 3:表示冒泡階段

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>DOM</title>
      <style>
         #box1 {
			width:300px;
            height:300px;
            background-color:skyblue;
          }
         #box2 {
			width:200px;
            height:200px;
            background-color:orange;
          }
          #box2 {
			width:100px;
            height:100px;
            background-color:tomato;
          }
      </style>
  </head>
  <body>
      <div id="box1">
          <div id="box2">
          	 <div id="box3">   
      	  	 </div> 
      	  </div>    
      </div>
      <script>
          const box1 = document.getElemenetById("box1")
          const box2 = document.getElemenetById("box2")
          const box3 = document.getElemenetById("box3")
          
          box1.addEventListener("click",(event)=>{
              alert(1)
          },true)
          box2.addEventListener("click",(event)=>{
              alert(2)
          },true)
          box3.addEventListener("click",(event)=>{
              alert(3)
          },true)
      </script>
  </body>
</html>



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 危機感 距離上一次找工作面試已經過去快2年了,那時候正值疫情肆虐,雖然還未感受到“寒潮來臨”的苗頭,但最終還是成功通過了幾輪面試,順利簽約。在目前公司待了2年了,在大環境的影響下,沒有加薪、沒有年終(這個真的很傷)、各種項目混亂、技術快停滯不前,年末又要過一年了,又離35進一步了,終危機感又來了,不 ...
  • 寫在前面 就在這周三,無意間我在掘金刷到一篇文章,讓我這個35歲的單身老狗又次相信了愛情,而且相信真的會有那種所謂的緣分和相濡以沫、雙向奔赴的愛情。 我又相信了愛情 文中男主是在掘金相親角成功的找到了另一半,而順利結婚,打動我的應該是女主的真誠吧,或許應該說那應該是我最嚮往的愛情,如下文中描述: 簡 ...
  • 在Go語言中,我們通常會遇到兩種主要的方式來處理和操作字元串:使用fmt.Sprintf函數和string.Builder類型。儘管兩者都可以實現字元串的格式化和連接,但它們在性能和用法上有一些關鍵區別。 1. fmt.Sprintf fmt.Sprintf是一個函數,它根據提供的格式化字元串和參數 ...
  • 在多線程編程中,如果每個線程的運行不是完全獨立的。那麼,一個線程執行到某個時刻需要知道其他線程發生了什麼。嗯,這就是所謂線程同步。同步事件對象(XXXEvent)有兩種行為: 1、等待。線程在此時會暫停運行,等待其他線程發出信號才繼續(等你約); 2、發出信號。當前線程發出信號,其他正在等待線程收到 ...
  • 思想 DAO(Data Access Object)數據訪問對象,是我們在做結構化資料庫訪問的時候傳輸的對象,通過這個對象我們可以與資料庫中的表建立映射關係 DTO(Data Transfer Object)是我們在與前端進行數據交換時傳遞的對象 為什麼需要設置這這兩種對象呢? 為了數據安全 如果我 ...
  • 1、Bring Up流程 SOC (System on a Chip) bring-up是一個複雜的過程,涉及到硬體、固件和軟體的集成和驗證,以下是一個基於BROM,SPL,UBOOT和Linux的啟動流程的概述: BROM (Boot Read-Only Memory)啟動:啟動的最初階段,在這個 ...
  • Android Studio簡單還原微信ui 上一期完成內容(前情提要) 上次我們簡單地實現了微信的幾個初始界面,並且在聯繫人頁面通過recycleview添加了許多的view 目標 建立在上次的基礎上,我們來擴展聯繫人界面的功能,給每一個view添加一個點擊功能,讓其可以跳轉到另一個activit ...
  • 本章將探討 HarmonyOS 的高級特性,包括分散式能力、安全機制和性能優化。這些特性可以幫助你構建更強大、更安全、更高效的應用。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...