真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

JavaScript中事件流的示例分析

小編給大家分享一下JavaScript中事件流的示例分析,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

目前創(chuàng)新互聯(lián)已為近千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、雅安服務(wù)器托管、網(wǎng)站運(yùn)營(yíng)、企業(yè)網(wǎng)站設(shè)計(jì)、新田網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶(hù)導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶(hù)和合作伙伴齊心協(xié)力一起成長(zhǎng),共同發(fā)展。

關(guān)于 JavaScript 的事件流

  • 前言

  • 正文

    • 甚麼是事件?

    • 甚麼是事件流?

    • 事件冒泡 vs 事件捕獲

    • DOM事件分級(jí)

    • DOM0 事件

    • DOM2 事件

  • 結(jié)語(yǔ)

前言

再看這篇前,建議各位可以先去看看 關(guān)於 JavaScript 事件循環(huán),當(dāng)然對(duì)於已經(jīng)了解的人們就不用啦!這篇要討論的是 js 的事件流。

正文

都知道, 當(dāng)我們?cè)诰W(wǎng)頁(yè)上做進(jìn)行某些類(lèi)型的操作,例如點(diǎn)擊,滑動(dòng)等等,都會(huì)觸發(fā)一些相應(yīng)的事件。也知道,整個(gè)網(wǎng)頁(yè)的構(gòu)成其實(shí)會(huì)被瀏覽器解析成一棵 DOM 樹(shù)。而當(dāng)一個(gè)節(jié)點(diǎn)產(chǎn)生一個(gè)事件時(shí),該事件會(huì)在該節(jié)點(diǎn)與根結(jié)點(diǎn)之間按一定順序傳播,而這個(gè)傳播路徑上經(jīng)過(guò)的所有結(jié)點(diǎn)都會(huì)接收到該事件,這一整個(gè)過(guò)程就稱(chēng)為DOM事件流。

甚麼是事件?

js 與 html 之間的交互其實(shí)就是通過(guò)「事件」實(shí)現(xiàn)的。舉凡用戶(hù)對(duì)網(wǎng)頁(yè)的點(diǎn)擊,選定,滑動(dòng)等等,在 js 的世界中全部都是事件。

而對(duì)於事件,當(dāng)事件發(fā)生就要有響應(yīng),在 js 中,所謂的響應(yīng)就是監(jiān)聽(tīng)器。就像觀察者模式一樣,事件就是我們的 subject,而當(dāng)事件發(fā)生時(shí),就要通知對(duì)應(yīng)該事件(subject)的所有監(jiān)聽(tīng)器(listener)去執(zhí)行相對(duì)的響應(yīng)。

甚麼是事件流?

事件流描述的就是從頁(yè)面中接收事件的順序。主要分為以下兩種:

  • IE 的事件冒泡

  • Netscape 的事件捕獲

事件冒泡 vs 事件捕獲

IE 提出的事件流模型是事件冒泡的,就是從下到上,從目標(biāo)觸發(fā)的元素逐級(jí)向上傳播,一直到 document 對(duì)象。
JavaScript中事件流的示例分析

Netscape 提出的事件流模型則是事件捕獲的,與事件冒泡相反,是從上到下的,即從 document 對(duì)象逐級(jí)向下傳播到目標(biāo)對(duì)象。
JavaScript中事件流的示例分析

以上是 DOM0 標(biāo)準(zhǔn)下的事件流機(jī)制。後來(lái) ECMAScript 在 DOM2 中對(duì)事件流做了進(jìn)一步的規(guī)範(fàn)。在 DOM2 中,一個(gè)事件所包含的事件流分為以下 3 個(gè)階段:

  1. 事件捕獲階段(capture)

  2. 目標(biāo)階段(target)

  3. 事件冒泡階段(bubble)
    JavaScript中事件流的示例分析

DOM事件分級(jí)

DOM節(jié)點(diǎn)有了事件發(fā)生,當(dāng)然就需要做相應(yīng)的處裡,而 DOM 事件又分為 4 個(gè)等級(jí),如下:
JavaScript中事件流的示例分析
其中,比較重要的是 DOM0/DOM2,所以下面著重介紹。

DOM0 事件

DOM0 級(jí)事件主要有兩種實(shí)現(xiàn)方式,第一種是內(nèi)聯(lián)模型,就是直接將函數(shù)名作為 html 標(biāo)籤中事件屬性的屬性值。如下:

// js code// eventDOM.jsfunction btnClick() {
    console.log('Hello World')}

    
        eventDOM demo
    
    
        

             

但是內(nèi)聯(lián)模型有明顯的缺點(diǎn),就是它違反了 W3C 對(duì)內(nèi)容(html)和行為(js)分離的要求。所以有了第二種,就是腳本模型(動(dòng)態(tài)綁定模型)。具體作法是通過(guò) js 腳本選定特定 DOM 節(jié)點(diǎn),然後對(duì)該節(jié)點(diǎn)添加事件屬性以及屬性值。如下:

// js code// eventDOM.jslet btn = document.getElementById('btn')let btnClick = function() {
    console.log('Hello World')}btn.onclick = btnClick

    
        eventDOM demo
    
    
        

             

點(diǎn)擊會(huì)出現(xiàn) Hello World,沒(méi)問(wèn)題。但腳本模型一樣有缺點(diǎn),基於上面的 html 代碼,添加一點(diǎn) js,如下:

// js code// eventDOM.jslet btn = document.getElementById('btn')let btnClick = function() {
    console.log('Hello World')}let btnClick2 = function() {
    console.log('Hello World again')}btn.onclick = btnClick
btn.onclick = btnClick2

我們發(fā)現(xiàn),現(xiàn)在點(diǎn)擊只會(huì)出現(xiàn) Hello World again。所以腳本模型只允許一個(gè)節(jié)點(diǎn)添加一次同類(lèi)型事件,後面的會(huì)覆蓋前面的。

最後讓我們?cè)賮?lái)看一個(gè)有趣的例子:


    
        eventDOM demo
    
    
        
            btn3            
                btn2                
                    btn1                

            

        

             
// js code// eventDOM.jslet btn1 = document.getElementById('btn1')let btn2 = document.getElementById('btn2')let btn3 = document.getElementById('btn3')btn1.onclick = function() {
    console.log('1')}btn2.onclick = function() {
    console.log('2')}btn3.onclick = function() {
    console.log('3')}

JavaScript中事件流的示例分析

當(dāng)我們點(diǎn)擊 btn3 時(shí),輸出如下:
JavaScript中事件流的示例分析
符合預(yù)期,但如果我們點(diǎn)擊 btn1 的話(huà)呢?輸出如下:
JavaScript中事件流的示例分析
這就有點(diǎn)奇怪了,明明我們?yōu)?btn1 只添加了一個(gè)監(jiān)聽(tīng)器阿,為什麼連同 btn2,btn3 的一起加上去了的感覺(jué)?原因就是因?yàn)?,雖然 DOM0 有 IE 提出的事件冒泡以及 Netscape 提出的事件捕獲兩種模型,但其實(shí) DOM0 只支持事件冒泡。所以,點(diǎn)擊 btn1 的事件流如下:
JavaScript中事件流的示例分析
也就是事件流也經(jīng)過(guò)了 btn2,btn3,所以才會(huì)觸發(fā)他們的事件處理。但是顯然,這並不是我們想要的結(jié)果。

DOM2 事件

進(jìn)一步規(guī)範(fàn)後,有了 DOM2級(jí)的事件處理程序。其中定義了兩個(gè)方法:

  • addEventListener() 添加事件監(jiān)聽(tīng)器

  • removeEventListener() 移除事件監(jiān)聽(tīng)器

這兩個(gè)函數(shù)均有三個(gè)參數(shù)如下表:

參數(shù)類(lèi)型描述
eventString監(jiān)聽(tīng)的事件名稱(chēng),比如’click’。注意這邊都不需要"on"
callbackfunction觸發(fā)事件所要執(zhí)行的回調(diào)函數(shù)
useCaptureBoolean(default:false)事件是否在捕獲階段進(jìn)行處理

DOM2 中就可以對(duì)同一個(gè)節(jié)點(diǎn)綁定兩個(gè)以上的同類(lèi)型事件監(jiān)聽(tīng)器了,看看下面例子:


    
        eventDOM demo
    
    
        

             
// js code// eventDOM.jslet btn = document.getElementById('btn')function hello() {
    console.log("Hello World")}function helloAgain() {
    console.log("Hello World again")}btn.addEventListener('click', hello, false)btn.addEventListener('click', helloAgain, false)

輸出如下:
JavaScript中事件流的示例分析 解決了 DOM0 只能綁定一個(gè)同類(lèi)型事件監(jiān)聽(tīng)器的缺點(diǎn)啦!值得注意的是,如果綁定同樣的監(jiān)聽(tīng)器兩次以上,仍然只會(huì)觸發(fā)一次。

再回到上面三個(gè) p 的那個(gè)例子進(jìn)行改寫(xiě)如下:


    
        eventDOM demo
    
    
        
            btn3            
                btn2                
                    btn1                

            

        

             
// js code// eventDOM.jslet btn1 = document.getElementById('btn1')let btn2 = document.getElementById('btn2')let btn3 = document.getElementById('btn3')btn1.addEventListener('click', function() {
    console.log('1')}, true)btn2.addEventListener('click', function() {
    console.log('2')}, true)btn3.addEventListener('click', function() {
    console.log('3')}, true)

注意,這邊我們把 addEventListener 的第三個(gè)參數(shù)設(shè)為 true,也就是事件會(huì)在捕獲階段觸發(fā)。點(diǎn)擊 btn1 輸出如下:
JavaScript中事件流的示例分析
看到順序與 DOM0 的順序反過(guò)來(lái)了。首先最外層(btn3)的節(jié)點(diǎn)先被觸發(fā)了,而因?yàn)榈谌齻€(gè)參數(shù)被設(shè)為 true,事件會(huì)在捕獲階段就被處理,所以輸出才會(huì)是 3,2,1。如果都是 false,就會(huì)是 1,2,3。

可見(jiàn) DOM2 的事件處理機(jī)制有了更彈性的操作空間。我們也可以在不同階段綁定事件監(jiān)聽(tīng)器,看看下面例子:

// js code// eventDOM.jsbtn1.addEventListener('click',function() {
    console.log('btn1 capture')}, true)btn1.addEventListener('click',function() {
    console.log('btn1 bubble')}, false)btn2.addEventListener('click',function() {
    console.log('btn2 capture')}, true)btn2.addEventListener('click',function() {
    console.log('btn2 bubble')}, false)btn3.addEventListener('click',function() {
    console.log('btn3 capture')}, true)btn3.addEventListener('click',function() {
    console.log('btn3 bubble')}, false)

點(diǎn)擊 btn1 輸出如下:
JavaScript中事件流的示例分析

改變一下順序,如下:

// js code// eventDOM.jsbtn1.addEventListener('click',function() {
    console.log('btn1 bubble')}, false)btn1.addEventListener('click',function() {
    console.log('btn1 capture')}, true)btn2.addEventListener('click',function() {
    console.log('btn2 bubble')}, false)btn2.addEventListener('click',function() {
    console.log('btn2 capture')}, true)btn3.addEventListener('click',function() {
    console.log('btn3 bubble')}, false)btn3.addEventListener('click',function() {
    console.log('btn3 capture')}, true)

點(diǎn)擊 btn1 輸出如下:
JavaScript中事件流的示例分析
注意 btn1 的輸出。雖然捕獲階段先發(fā)生了,但是因?yàn)?btn1 本身就是目標(biāo)節(jié)點(diǎn),所以在這種情況下,總結(jié)出規(guī)律:在目標(biāo)元素上不區(qū)分冒泡還是捕獲,是根據(jù)腳本中的順序來(lái)執(zhí)行。

有時(shí)候,我們希望對(duì)於某節(jié)點(diǎn),不要再經(jīng)過(guò)冒泡階段了,DOM2 也提供了相應(yīng)函數(shù),stopPropagation

// js code// eventDOM.jsbtn1.addEventListener('click',function() {
    console.log('btn1 capture')}, true)btn1.addEventListener('click',function() {
    console.log('btn1 bubble')}, false)btn2.addEventListener('click',function(e) {
    e.stopPropagation()
    console.log('btn2 capture')}, true)btn2.addEventListener('click',function() {
    console.log('btn2 bubble')}, false)btn3.addEventListener('click',function() {
    console.log('btn3 capture')}, true)btn3.addEventListener('click',function() {
    console.log('btn3 bubble')}, false)

點(diǎn)擊 btn1 輸出如下:
JavaScript中事件流的示例分析
可以看到,因?yàn)槲覀冊(cè)?btn2 的捕獲階段就阻止了 btn2 的冒泡階段,所以 btn2 在捕獲後就不再繼續(xù)執(zhí)行下去,確保不會(huì)冒泡,事件流如下:
JavaScript中事件流的示例分析

加強(qiáng)一下前面提到的知識(shí)點(diǎn),如果是在 btn1 阻止冒泡,會(huì)變成怎樣呢?

// js code// eventDOM.jsbtn1.addEventListener('click',function(e) {
    e.stopPropagation()
    console.log('btn1 capture')}, true)btn1.addEventListener('click',function() {
    console.log('btn1 bubble')}, false)btn2.addEventListener('click',function() {
    console.log('btn2 capture')}, true)btn2.addEventListener('click',function() {
    console.log('btn2 bubble')}, false)btn3.addEventListener('click',function() {
    console.log('btn3 capture')}, true)btn3.addEventListener('click',function() {
    console.log('btn3 bubble')}, false)

點(diǎn)擊 btn1 輸出如下:
JavaScript中事件流的示例分析
雖然我們對(duì) btn1 阻止了冒泡,但是為什麼還是輸出了 btn bubble呢?原因就是前面提到了,目標(biāo)節(jié)點(diǎn)不區(qū)分 捕獲/冒泡 階段,但是後面也就不會(huì)繼續(xù)冒泡了,算是個(gè)比較特殊的情況,可以稍微留意下。

以上是“JavaScript中事件流的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!


網(wǎng)頁(yè)標(biāo)題:JavaScript中事件流的示例分析
路徑分享:http://weahome.cn/article/iieoip.html

其他資訊

在線咨詢(xún)

微信咨詢(xún)

電話(huà)咨詢(xún)

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部