javascript中事件流的示例分析-mile米乐体育
小编给大家分享一下javascript中事件流的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!
关于 javascript 的事件流
前言
正文
甚麼是事件?
甚麼是事件流?
事件冒泡 vs 事件捕獲
dom事件分級
dom0 事件
dom2 事件
結語
前言
再看這篇前,建議各位可以先去看看 關於 javascript 事件循環,當然對於已經了解的人們就不用啦!這篇要討論的是 js 的事件流。
正文
都知道, 當我們在網頁上做進行某些類型的操作,例如點擊,滑動等等,都會觸發一些相應的事件。也知道,整個網頁的構成其實會被瀏覽器解析成一棵 dom 樹。而當一個節點產生一個事件時,該事件會在該節點與根結點之間按一定順序傳播,而這個傳播路徑上經過的所有結點都會接收到該事件,這一整個過程就稱為dom事件流。
甚麼是事件?
js 與 html 之間的交互其實就是通過「事件」實現的。舉凡用戶對網頁的點擊,選定,滑動等等,在 js 的世界中全部都是事件。
而對於事件,當事件發生就要有響應,在 js 中,所謂的響應就是監聽器。就像觀察者模式一樣,事件就是我們的 subject,而當事件發生時,就要通知對應該事件(subject)的所有監聽器(listener)去執行相對的響應。
甚麼是事件流?
事件流描述的就是從頁面中接收事件的順序。主要分為以下兩種:
ie 的事件冒泡
netscape 的事件捕獲
事件冒泡 vs 事件捕獲
ie 提出的事件流模型是事件冒泡的,就是從下到上,從目標觸發的元素逐級向上傳播,一直到 document 對象。
netscape 提出的事件流模型則是事件捕獲的,與事件冒泡相反,是從上到下的,即從 document 對象逐級向下傳播到目標對象。
以上是 dom0 標準下的事件流機制。後來 ecmascript 在 dom2 中對事件流做了進一步的規範。在 dom2 中,一個事件所包含的事件流分為以下 3 個階段:
事件捕獲階段(capture)
目標階段(target)
事件冒泡階段(bubble)
dom事件分級
dom節點有了事件發生,當然就需要做相應的處裡,而 dom 事件又分為 4 個等級,如下:
dom0 事件
dom0 級事件主要有兩種實現方式,第一種是內聯模型,就是直接將函數名作為 html 標籤中事件屬性的屬性值。如下:
//jscode//eventdom.jsfunctionbtnclick(){ console.log('helloworld')}
eventdomdemo
但是內聯模型有明顯的缺點,就是它違反了 w3c 對內容(html)和行為(js)分離的要求。所以有了第二種,就是腳本模型(動態綁定模型)。具體作法是通過 js 腳本選定特定 dom 節點,然後對該節點添加事件屬性以及屬性值。如下:
//jscode//eventdom.jsletbtn=document.getelementbyid('btn')letbtnclick=function(){ console.log('helloworld')}btn.onclick=btnclick
eventdomdemo
點擊會出現 hello world
,沒問題。但腳本模型一樣有缺點,基於上面的 html 代碼,添加一點 js,如下:
//jscode//eventdom.jsletbtn=document.getelementbyid('btn')letbtnclick=function(){ console.log('helloworld')}letbtnclick2=function(){ console.log('helloworldagain')}btn.onclick=btnclick btn.onclick=btnclick2
我們發現,現在點擊只會出現 hello world again
。所以腳本模型只允許一個節點添加一次同類型事件,後面的會覆蓋前面的。
最後讓我們再來看一個有趣的例子:
eventdomdemo btn3 btn2 btn1
//jscode//eventdom.jsletbtn1=document.getelementbyid('btn1')letbtn2=document.getelementbyid('btn2')letbtn3=document.getelementbyid('btn3')btn1.onclick=function(){ console.log('1')}btn2.onclick=function(){ console.log('2')}btn3.onclick=function(){ console.log('3')}
當我們點擊 btn3 時,輸出如下:
dom2 事件
進一步規範後,有了 dom2級的事件處理程序。其中定義了兩個方法:
addeventlistener() 添加事件監聽器
removeeventlistener() 移除事件監聽器
這兩個函數均有三個參數如下表:
參數 | 類型 | 描述 |
---|---|---|
event | string | 監聽的事件名稱,比如’click’。注意這邊都不需要"on" |
callback | function | 觸發事件所要執行的回調函數 |
usecapture | boolean(default:false) | 事件是否在捕獲階段進行處理 |
dom2 中就可以對同一個節點綁定兩個以上的同類型事件監聽器了,看看下面例子:
eventdomdemo
//jscode//eventdom.jsletbtn=document.getelementbyid('btn')functionhello(){ console.log("helloworld")}functionhelloagain(){ console.log("helloworldagain")}btn.addeventlistener('click',hello,false)btn.addeventlistener('click',helloagain,false)
輸出如下:
再回到上面三個 p 的那個例子進行改寫如下:
eventdomdemo btn3 btn2 btn1
//jscode//eventdom.jsletbtn1=document.getelementbyid('btn1')letbtn2=document.getelementbyid('btn2')letbtn3=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
的第三個參數設為 true,也就是事件會在捕獲階段觸發。點擊 btn1 輸出如下:
可見 dom2 的事件處理機制有了更彈性的操作空間。我們也可以在不同階段綁定事件監聽器,看看下面例子:
//jscode//eventdom.jsbtn1.addeventlistener('click',function(){ console.log('btn1capture')},true)btn1.addeventlistener('click',function(){ console.log('btn1bubble')},false)btn2.addeventlistener('click',function(){ console.log('btn2capture')},true)btn2.addeventlistener('click',function(){ console.log('btn2bubble')},false)btn3.addeventlistener('click',function(){ console.log('btn3capture')},true)btn3.addeventlistener('click',function(){ console.log('btn3bubble')},false)
點擊 btn1 輸出如下:
改變一下順序,如下:
//jscode//eventdom.jsbtn1.addeventlistener('click',function(){ console.log('btn1bubble')},false)btn1.addeventlistener('click',function(){ console.log('btn1capture')},true)btn2.addeventlistener('click',function(){ console.log('btn2bubble')},false)btn2.addeventlistener('click',function(){ console.log('btn2capture')},true)btn3.addeventlistener('click',function(){ console.log('btn3bubble')},false)btn3.addeventlistener('click',function(){ console.log('btn3capture')},true)
點擊 btn1 輸出如下:
有時候,我們希望對於某節點,不要再經過冒泡階段了,dom2 也提供了相應函數,stoppropagation
。
//jscode//eventdom.jsbtn1.addeventlistener('click',function(){ console.log('btn1capture')},true)btn1.addeventlistener('click',function(){ console.log('btn1bubble')},false)btn2.addeventlistener('click',function(e){ e.stoppropagation() console.log('btn2capture')},true)btn2.addeventlistener('click',function(){ console.log('btn2bubble')},false)btn3.addeventlistener('click',function(){ console.log('btn3capture')},true)btn3.addeventlistener('click',function(){ console.log('btn3bubble')},false)
點擊 btn1 輸出如下:
加強一下前面提到的知識點,如果是在 btn1 阻止冒泡,會變成怎樣呢?
//jscode//eventdom.jsbtn1.addeventlistener('click',function(e){ e.stoppropagation() console.log('btn1capture')},true)btn1.addeventlistener('click',function(){ console.log('btn1bubble')},false)btn2.addeventlistener('click',function(){ console.log('btn2capture')},true)btn2.addeventlistener('click',function(){ console.log('btn2bubble')},false)btn3.addeventlistener('click',function(){ console.log('btn3capture')},true)btn3.addeventlistener('click',function(){ console.log('btn3bubble')},false)
點擊 btn1 輸出如下:btn bubble
呢?原因就是前面提到了,目標節點不區分 捕獲/冒泡 階段,但是後面也就不會繼續冒泡了,算是個比較特殊的情況,可以稍微留意下。
以上是“javascript中事件流的示例分析”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注恰卡编程网行业资讯频道!