亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

基于事件冒泡、事件捕獲和事件委托詳解

 更新時間:2021年07月08日 11:41:28   作者:晏子楚  
這篇文章主要介紹了事件冒泡、事件捕獲和事件委托,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

事件冒泡、事件捕獲和事件委托

在javascript里,事件委托是很重要的一個東西,事件委托依靠的就是事件冒泡和捕獲的機制,我先來解釋一下事件冒泡和事件捕獲:

事件冒泡會從當前觸發(fā)的事件目標一級一級往上傳遞,依次觸發(fā),直到document為止。

事件捕獲會從document開始觸發(fā),一級一級往下傳遞,依次觸發(fā),直到真正事件目標為止。

這么說是不是很抽象,其實就像我敲擊了一下鍵盤,我在敲擊鍵盤的同時,我是不是也敲擊了這臺電腦,我寫個例子大家就明白了:

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8">
        <title></title>
    </head>
    <style type="text/css">
        #box1 { width: 300px; height: 300px; background: blueviolet; }
        #box2 { width: 200px; height: 200px; background: aquamarine; }
        #box3 { width: 100px; height: 100px; background: tomato; }
        div { overflow: hidden; margin: 50px auto; }
    </style>
    <body>
        <div id="box1">
            <div id="box2">
                <div id="box3"></div>
            </div>
        </div>
        <script>
            function sayBox3() {
                console.log('你點了最里面的box');
            }
            function sayBox2() {
                console.log('你點了最中間的box');
            }
            function sayBox1() {
                console.log('你點了最外面的box');
            }
            // 事件監(jiān)聽,第三個參數是布爾值,默認false,false是事件冒泡,true是事件捕獲
            document.getElementById('box3').addEventListener('click', sayBox3, false);
            document.getElementById('box2').addEventListener('click', sayBox2, false);
            document.getElementById('box1').addEventListener('click', sayBox1, false);
        </script>
    </body>
</html>

我們畫了三個box,結構是父子關系,分別綁定了打印事件,現在我們來點擊最中間的紅色box:

我們發(fā)現,我們僅僅是點擊了紅色的box,但是綠色和紫色的box也被觸發(fā)了打印事件,觸犯順序是 紅色>綠色>紫色,這種現象就是事件冒泡了。

我們再試試事件捕獲,把上面代碼里監(jiān)聽事件的第三個參數改為true,然后點擊紅色的box:

我們還是只點擊最中間的紅色box,和上一次一樣,也是三個box都觸發(fā)了事件,但是順序反過來了,紫色>綠色>紅色,這種現象稱為事件捕獲。

通過上面的例子,應該很容易就理解了事件冒泡和事件捕獲,我們平時都是默認冒泡的,冒泡是一直冒到document根文檔為止。

現在來談談事件委托,事件委托又稱之為事件代理,我們通過一個通俗的例子來解釋:

有三個同事預計會在周一收到快遞,為了簽收快遞,有兩種辦法:1.三個人在公司門口等快遞;2.委托給前臺MM代為簽收?,F實當中,我們大都采用委托的方案(公司也不會容忍那么多員工站在門口就為了等快遞)。前臺MM收到快遞后,她會判斷收件人是誰,然后按照收件人的要求簽收,甚至代為付款。這種方案還有一個優(yōu)勢,那就是即使公司里來了新員工(不管多少),前臺MM也會在收到寄給新員工的快遞后核實并代為簽收(可以給暫時不存在的節(jié)點也綁定上事件)。

我們再舉另一個例子:

現在有一個ul,ul里又有100個li,我想給這100個li都綁定一個點擊事件,我們一般可以通過for循環(huán)來綁定,但是要是有1000個li呢? 為了提高效率和速度,所以我們這時可以采用事件委托,只給ul綁定一個事件,根據事件冒泡的規(guī)則,只要你點了ul里的每一個li,都會觸發(fā)ul的綁定事件,我們在ul綁定事件的函數里通過一些判斷,就可以給這100li都觸發(fā)點擊事件了。

具體怎么實現,看代碼:

// 這里不講IE,結尾再說
function clickLi() {
    alert('你點擊了li');
}
document.getElementById('isUl').addEventListener('click', function(event) {
    // 每一個函數內都有一個event事件對象,它有一個target屬性,指向事件源
    var src = event.target;
    // 我們判斷如果target事件源的節(jié)點名字是li,那就執(zhí)行這個函數
    // target里面的屬性是非常多的,id名、class名、節(jié)點名等等都可以取到
    if(src.nodeName.toLowerCase() == 'li') {
       clickLi() ;
    }
});

這樣我們就通過給ul綁定一個點擊事件,讓所有的li都觸發(fā)了函數。

那如果想給不同的li綁定不同的函數怎么辦?

假設有3個li,我們先寫3個不同的函數,再給3個li設置不同的id名,通過判斷id名是不是就能給不同的li綁定不同的函數啦:

<body>
    <ul id="isUl">
        <li id="li01">1</li>
        <li id="li02">2</li>
        <li id="li03">3</li>
    </ul>
    <script>
        function clickLi01() {
            alert('你點擊了第1個li');
        }
        function clickLi02() {
            alert('你點擊了第2個li');
        }
        function clickLi03() {
            alert('你點擊了第3個li');
        }
        document.getElementById('isUl').addEventListener('click', function(event) {
            var srcID = event.target.id;
            if(srcID == 'li01'){
                clickLi01();
            }else if(srcID == 'li02') {
                clickLi02();
            }else if(srcID == 'li03') {
                clickLi03();
            }
        });
    </script>
</body>

這就是所謂的事件委托,通過監(jiān)聽一個父元素,來給不同的子元素綁定事件,減少監(jiān)聽次數,從而提升速度。

那么,能不能阻止元素的事件冒泡呢,答案是可以的。

一開始那個例子,假如我們真的只想點擊最里面的那個紅色box,不想另外兩個box的事件被觸發(fā),我們可以在給紅色box綁定事件的函數里這么寫:

function sayBox3(event) {
    // 阻止冒泡
    event.stopPropagation();
    console.log('你點了最里面的box');
}
document.getElementById('box3').addEventListener('click', sayBox3, false);

這樣我們再點擊紅色的box,那就只會觸發(fā)它本身的事件啦。

那阻止冒泡有沒有實際用途呢?答案是有的,我們看這個例子:

這是一個模態(tài)框,現在的需求是當我們點擊紅色的按鈕需要跳轉頁面,然后點擊白色的對話框不需要任何反應,點其它任何地方就關閉這個模態(tài)框。

這里就需要用到阻止冒泡了,紅色的按鈕是白色對話框的子元素,白色對話框又是這整個模態(tài)框的子元素,我們給模態(tài)框加上一個點擊事件關閉,然后給紅色的按鈕加上一個點擊事件跳轉,這時就產生了一個問題,只要點擊白色的對話框,由于冒泡機制,這個模態(tài)框也會關閉,實際上我們并不想點擊白色的對話框有任何反應,這時我們就給這個白色的對話框綁定一個點擊事件,函數里寫上event.stopPropagation();,這樣就OK了。

關于IE

老版本的IE存在兼容問題,根本不支持addEventListener()的添加事件和removeEventListener()的刪除事件,它有自己的監(jiān)聽方法:

// 添加事件,事件流固定為冒泡
attachEvent(事件名,事件處理函數)
// 刪除事件
detachEvent(事件名,事件處理函數)

還有IE里的事件對象是window.event,事件源是srcElement,阻止冒泡寫法也不一樣:

function() {
    // IE里阻止冒泡
    window.event.cancelBubble = true;
    // IE里獲取事件源的id
    var srcID = window.event.srcElement.id;
}
function(event) {
    // 非IE里阻止冒泡
    event.stopPropagation();
    // 非IE里獲取事件源的id
    var srcID = event.target.id;
}

補充

關于js的瀏覽器兼容問題,一般用能力檢測來解決,if(){}else{}

我們平時工作一般都是用jquery,IE這些特殊情況早就幫我們做好兼容啦。

jquery在1.7的版本之后,最流行的事件監(jiān)聽方法是$(元素).on(事件名,執(zhí)行函數),它還有一種事件委托的寫法$(委托給哪個元素).on(事件名,被委托的元素,執(zhí)行函數)

最后說一點,如果元素被阻止冒泡了,千萬別去用事件委托的方式監(jiān)聽事件,因為事件委托的原理是利用事件冒泡,當冒泡被阻止,就無法監(jiān)聽了。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關文章

  • js實現簡單的前端分頁效果

    js實現簡單的前端分頁效果

    這篇文章主要為大家詳細介紹了js實現簡單的前端分頁效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • 簡單實現IONIC購物車功能

    簡單實現IONIC購物車功能

    這篇文章主要為大家詳細介紹了IONIC簡易購物車的實現代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-01-01
  • Java/JS獲取flash高寬的具體方法

    Java/JS獲取flash高寬的具體方法

    本文為大家詳細介紹下使用Java/JS如何獲取flash的高寬,下面有個不錯的示例,感興趣的朋友可以參考下,希望對大家有所幫助
    2013-12-12
  • 淺析JavaScriptSerializer類的序列化與反序列化

    淺析JavaScriptSerializer類的序列化與反序列化

    本篇文章主要介紹了JavaScriptSerializer類 對象序列化與反序列化的方法、屬性以及實例代碼,有需要的朋友可以參考一下
    2016-11-11
  • javascript實現查找數組中最大值方法匯總

    javascript實現查找數組中最大值方法匯總

    本文給大家匯總了一下使用javascript實現查找數組中最大最小值的一些方法,非常的簡單實用,有需要的小伙伴可以來參考下。
    2016-02-02
  • javascript針對cookie的基本操作實例詳解

    javascript針對cookie的基本操作實例詳解

    這篇文章主要介紹了javascript針對cookie的基本操作,結合實例形式較為詳細的分析了JavaScript操作cookie的基本技巧,并給出了一個JavaScript操作cookie的完整類,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-11-11
  • Smartour 讓網頁導覽變得更簡單(推薦)

    Smartour 讓網頁導覽變得更簡單(推薦)

    這篇文章主要介紹了Smartour 讓網頁導覽變得更簡單(推薦),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-07-07
  • 最新評論