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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...