ホームページ ウェブフロントエンド jsチュートリアル JS のイベント バブリング メカニズムを 1 つの記事で理解する

JS のイベント バブリング メカニズムを 1 つの記事で理解する

Aug 04, 2022 pm 08:37 PM
javascript イベントバブリング

この記事ではイベント バブリングについて説明し、JS のイベント バブリング メカニズムについて詳しく説明します。お役に立てば幸いです。

JS のイベント バブリング メカニズムを 1 つの記事で理解する

#1. イベント

ブラウザクライアントアプリケーションプラットフォームでは、基本的に生成されます。これはイベント駆動型です。つまり、イベントが発生すると、対応するアクションが実行されます。

ブラウザ イベントは、何かが起こったという信号を表します。事件の説明はこの記事の焦点では​​ありません。まだ理解していない友人は、Baidu で詳細を学ぶことができ、以下の内容をより深く理解するのに役立ちます。

2. 気泡の仕組み

#バブリングとは?

下の図を見ていただければわかると思いますが、泡は水底から始まり、深いところから浅いところ、そして上の方へと上がっていきます。上昇する途中で、泡はさまざまな深さの水中を通過します。

DOM ツリー全体。イベントは、DOM のルート ノードに渡されるまで、DOM ツリーの最下層から上に渡されます。 JS のイベント バブリング メカニズムを 1 つの記事で理解する

簡単なケース分析バブリングの原理を説明するための簡単な例を以下に示します。 HTML には 3 つの単純な DOM 要素があります: div1、div2、span、div2 には span が含まれ、div1 には div2 が含まれます。これらはすべて本体の下にあります:

	<div>
		<div>
			<span>This is a span.</span>
		</div>
	</div>
ログイン後にコピー
Interface professional as

、ボディがイベント イベントをキャプチャすると、イベントが発生した時刻とイベントをトリガーしたノード情報が出力されます:

<script>
	window.onload = function() {
		document.getElementById("body").addEventListener("click",eventHandler);
	}
	function eventHandler(event) {
		console.log("时间:"+new Date(event.timeStamp)+" 产生事件的节点:" + event.target.id +"  当前节点:"+event.currentTarget.id);
	}
</script>
ログイン後にコピー
クリックすると「これはスパンです」、div2、div1、およびbodyが順番に、

JS のイベント バブリング メカニズムを 1 つの記事で理解する#上記の結果の分析を出力します: 、または div のサブ要素 div2、span これらの要素がクリックされると、クリック イベントが生成され、本体がそれをキャプチャして、対応するイベント処理関数を呼び出します。水の中の泡が下から上に上昇するのと同じように、出来事も上昇します。

イベント配信の概略図は次のとおりです:

# 通常、イベント配信プロセス中にいくつかの情報が表示されます。イベントの構成要素:

イベントが発生した時刻 イベントが発生した場所 イベントの種類 イベントの現在のハンドラー その他の情報 JS のイベント バブリング メカニズムを 1 つの記事で理解する

完全な HTML コードは次のとおりです:

nbsp;html>



<script></script>
Insert title here

 
<script>
	window.onload = function() {
		document.getElementById("body").addEventListener("click",eventHandler);
	}
	function eventHandler(event) {
		console.log("时间:"+new Date(event.timeStamp)+" 产生事件的节点:" + event.target.id +"  当前节点:"+event.currentTarget.id);
	}
</script>
 


	<div>
		<div>
			<span>This is a span.</span>
		</div>
	</div>

ログイン後にコピー

b. 終了イベントのバブリング

JS のイベント バブリング メカニズムを 1 つの記事で理解する

次に、このようなイベントを実装したいと思います。関数では、div1をクリックすると「こんにちは、一番外側のdivです。」が表示され、div2をクリックすると「こんにちは、2層目のdivです。」が表示され、spanをクリックすると「こんにちは、私は2番目の層のdivです」と表示されます。 2 番目の div" が表示されます。span." これから、次の JavaScript フラグメントが得られます:

<script>
	window.onload = function() {
		document.getElementById("box1").addEventListener("click",function(event){
			alert("您好,我是最外层div。");
		});
		document.getElementById("box2").addEventListener("click",function(event){
			alert("您好,我是第二层div。");
		});
		document.getElementById("span").addEventListener("click",function(event){
			alert("您好,我是span。");
		});
	}
</script>
ログイン後にコピー

预期上述代码会单击span 的时候,会出来一个弹出框 "您好,我是span。" 是的,确实弹出了这样的对话框:

JS のイベント バブリング メカニズムを 1 つの記事で理解する

然而,不仅仅会产生这个对话框,当点击确定后,会依次弹出下列对话框:

JS のイベント バブリング メカニズムを 1 つの記事で理解する     JS のイベント バブリング メカニズムを 1 つの記事で理解する

这显然不是我们想要的! 我们希望的是点谁显示谁的信息而已。为什么会出现上述的情况呢? 原因就在于事件的冒泡,点击span的时候,span 会把产生的事件往上冒泡,作为父节点的div2 和 祖父节点的div1也会收到此事件,于是会做出事件响应,执行响应函数。现在问题是发现了,但是怎么解决呢?

方法一:我们来考虑一个形象一点的情况:水中的一个气泡正在从底部往上冒,而你现在在水中,不想让这个气泡往上冒,怎么办呢?——把它扎破!没了气泡,自然不会往上冒了。类似地,对某一个节点而言,如果不想它现在处理的事件继续往上冒泡的话,我们可以终止冒泡:

在相应的处理函数内,加入  event.stopPropagation()   ,终止事件的广播分发,这样事件停留在本节点,不会再往外传播了。修改上述的script片段:

<script>
	window.onload = function() {
		document.getElementById("box1").addEventListener("click",function(event){
			alert("您好,我是最外层div。");
			event.stopPropagation();
		});
		document.getElementById("box2").addEventListener("click",function(event){
			alert("您好,我是第二层div。");
			event.stopPropagation();
		});
		document.getElementById("span").addEventListener("click",function(event){
			alert("您好,我是span。");
			event.stopPropagation();
		});
	}
</script>
ログイン後にコピー

经过这样一段代码,点击不同元素会有不同的提示,不会出现弹出多个框的情况了。

方法二:事件包含最初触发事件的节点引用 和 当前处理事件节点的引用,那如果节点只处理自己触发的事件即可,不是自己产生的事件不处理。event.target 引用了产生此event对象的dom 节点,而event.currrentTarget 则引用了当前处理节点,我们可以通过这 两个target 是否相等。

            比如span 点击事件,产生一个event 事件对象,event.target 指向了span元素,span处理此事件时,event.currentTarget 指向的也是span元素,这时判断两者相等,则执行相应的处理函数。而事件传递给 div2 的时候,event.currentTarget变成 div2,这时候判断二者不相等,即事件不是div2 本身产生的,就不作响应处理逻辑。               

<script>
	window.onload = function() {
		document.getElementById("box1").addEventListener("click",function(event){
			if(event.target == event.currentTarget)
			{
			    alert("您好,我是最外层div。");
			}
		});
		document.getElementById("box2").addEventListener("click",function(event){
			if(event.target == event.currentTarget)
			{
				alert("您好,我是第二层div。");
			}
		});
		document.getElementById("span").addEventListener("click",function(event){
			if(event.target == event.currentTarget)
			{
			    alert("您好,我是span。");
				
			}
		});
	}
</script>
ログイン後にコピー

比较:

从事件传递上看:

  • 方法一在于取消事件冒泡,即当某些节点取消冒泡后,事件不会再传递;

  • 方法二在于不阻止冒泡,过滤需要处理的事件,事件处理后还会继续传递;

优缺点:

  • 方法一缺点:为了实现点击特定的元素显示对应的信息,方法一要求每个元素的子元素也必须终止事件的冒泡传递,即跟别的元素功能上强关联,这样的方法会很脆弱。比如,如果span 元素的处理函数没有执行冒泡终止,则事件会传到p2 上,这样会造成p2 的提示信息;

  • 方法二缺点:方法二为每一个元素都增加了事件监听处理函数,事件的处理逻辑都很相似,即都有判断 if(event.target == event.currentTarget),这样存在了很大的代码冗余,现在是三个元素还好,当有10几个,上百个又该怎么办呢?

还有就是为每一个元素都有处理函数,在一定程度上增加逻辑和代码的复杂度。

        我们再来分析一下方法二:方法二的原理是 元素收到事件后,判断事件是否符合要求,然后做相应的处理,然后事件继续冒泡往上传递;

既然事件是冒泡传递的,那可不可以让某个父节点统一处理事件,通过判断事件的发生地(即事件产生的节点),然后做出相应的处理呢?答案是可以的,下面通过给body 元素添加事件监听,然后通过判断event.target 然后对不同的target产生不同的行为。

        将方法二的代码重构一下:

<script>
	window.onload = function() {
		document.getElementById("body").addEventListener("click",eventPerformed);
	}
	function eventPerformed(event) {
		var target = event.target;
		switch (target.id) {
		case "span": 
			alert("您好,我是span。");
			break;
		case "div1":
			alert("您好,我是第二层div。");
			break;
		case "div2":
			alert("您好,我是最外层div。");
			break;
		}
	}
</script>
ログイン後にコピー

结果会是点击不同的元素,只弹出相符合的提示,不会有多余的提示。

p1 は応答ロジックを本体に委任し、対応するロジックを完成させる 対応するロジックを実装しない このモードは、いわゆるイベントデリゲーションです。
以下は概略図です:


JS のイベント バブリング メカニズムを 1 つの記事で理解する[関連する推奨事項:

JavaScript 学習チュートリアル

#]

以上がJS のイベント バブリング メカニズムを 1 つの記事で理解するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

イベントのバブリング メカニズムを理解する: 子要素をクリックすると親要素のイベントに影響を与えるのはなぜですか? イベントのバブリング メカニズムを理解する: 子要素をクリックすると親要素のイベントに影響を与えるのはなぜですか? Jan 13, 2024 pm 02:55 PM

イベントバブリングについて: 子要素をクリックすると親要素のイベントがトリガーされるのはなぜですか?イベントバブリングとは、入れ子になった要素構造において、子要素がイベントをトリガーすると、そのイベントがバブリングのように層ごとに親要素の層に最外層の親要素まで渡されることを意味します。このメカニズムにより、子要素のイベントが要素ツリー全体に伝播され、関連するすべての要素が順番にトリガーされます。イベントバブリングをより深く理解するために、具体的なコード例を見てみましょう。 HTML コード: <divid="parent&q"

jQuery .val() が失敗する理由と解決策 jQuery .val() が失敗する理由と解決策 Feb 20, 2024 am 09:06 AM

タイトル: jQuery.val() が失敗する理由と解決策 フロントエンド開発では DOM 要素の操作に jQuery がよく使われますが、フォーム要素の値の取得や設定には .val() メソッドが広く使われています。ただし、.val() メソッドが失敗し、フォーム要素の値を正しく取得または設定できない状況が発生することがあります。この記事では、.val() エラーの原因を調査し、対応する解決策を提供し、具体的なコード例を添付します。 1.原因分析.val()メソッド

イベントのバブリングが 2 回トリガーされるのはなぜですか? イベントのバブリングが 2 回トリガーされるのはなぜですか? Feb 22, 2024 am 09:06 AM

イベントのバブリングが 2 回トリガーされるのはなぜですか?イベント バブリング (イベント バブリング) とは、DOM において、要素がイベント (クリック イベントなど) をトリガーすると、イベントがその要素から親要素にバブルアップし、最後に最上位のドキュメント オブジェクトにバブルアップすることを意味します。イベント バブリングは DOM イベント モデルの一部であり、これにより開発者はイベント リスナーを親要素にバインドできるため、子要素がイベントをトリガーすると、バブリング メカニズムを通じてイベントをキャプチャして処理できます。ただし、開発者は、バブルして 2 回トリガーされるイベントに遭遇することがあります。

jsのクリックイベントが繰り返し実行できない理由 jsのクリックイベントが繰り返し実行できない理由 May 07, 2024 pm 06:36 PM

JavaScript のクリック イベントは、イベント バブリング メカニズムのため、繰り返し実行できません。この問題を解決するには、次の措置を講じることができます。 イベント キャプチャを使用する: イベントがバブルアップする前に起動するイベント リスナーを指定します。イベントの引き継ぎ: イベントのバブリングを停止するには、event.stopPropagation() を使用します。タイマーを使用します。しばらくしてからイベント リスナーを再度トリガーします。

vue のイベント修飾子はどのようなシナリオに使用できますか? vue のイベント修飾子はどのようなシナリオに使用できますか? May 09, 2024 pm 02:33 PM

Vue.js イベント修飾子は、次のような特定の動作を追加するために使用されます。 デフォルト動作の防止 (.prevent) イベント バブリングの停止 (.stop) ワンタイム イベント (.once) イベントのキャプチャ (.capture) パッシブ イベント リスニング (.passive) アダプティブ修飾子 (.self)キー修飾子 (.key)

イベント バブリング メカニズムが 2 回トリガーされるのはなぜですか? イベント バブリング メカニズムが 2 回トリガーされるのはなぜですか? Feb 25, 2024 am 09:24 AM

イベントのバブリングが 2 回連続して発生するのはなぜですか?イベント バブリングは Web 開発における重要な概念であり、ネストされた HTML 要素でイベントがトリガーされると、イベントが最も内側の要素から最も外側の要素までバブルアップすることを意味します。このプロセスは時々混乱を引き起こす可能性があり、よくある問題の 1 つは、イベントのバブリングが 2 回続けて発生することです。イベント バブリングが 2 回連続して発生する理由をよりよく理解するために、まずコード例を見てみましょう。

バブルアップしない JS イベントはどれですか? バブルアップしない JS イベントはどれですか? Feb 19, 2024 pm 09:56 PM

JS イベントでバブルが発生しない状況にはどのようなものがありますか?イベントバブリング(Event Bubble)とは、要素上でイベントが発生した後、最も内側の要素から最外側の要素に向かってDOMツリーに沿って上方向にイベントが送信されることを意味し、この送信方法をイベントバブリングと呼びます。ただし、すべてのイベントがバブル化できるわけではなく、イベントがバブル化しない特殊なケースもいくつかあります。この記事では、JavaScript でイベントがバブルアップしない状況を紹介します。 1. stopPropagati を使用する

バブリングイベントを防ぐ一般的な方法は何ですか? バブリングイベントを防ぐ一般的な方法は何ですか? Feb 19, 2024 pm 10:25 PM

バブリングイベントを防ぐために一般的に使用されるコマンドは何ですか? Web 開発では、イベントのバブリングを処理する必要がある状況によく遭遇します。クリック イベントなどのイベントが要素でトリガーされると、その親要素も同じイベントをトリガーします。イベント配信のこの動作は、イベント バブリングと呼ばれます。場合によっては、イベントがバブルアップするのを防ぎ、イベントが現在の要素でのみ発生し、上位の要素に渡されないようにしたい場合があります。これを実現するには、バブリング イベントを防ぐいくつかの一般的なディレクティブを使用できます。イベント.ストッププロパ

See all articles