view事件分發(fā)機(jī)制

   作為一名Android程序猿,相信你一定碰到過滑動(dòng)沖突這一問題,解決它的理論基礎(chǔ)就是要了解view的事件分發(fā)機(jī)制,本博客只是從大的方面分析事件分發(fā)機(jī)制,如果要深入研究它,建議大家結(jié)合系統(tǒng)源碼去進(jìn)一步分析事件分發(fā)機(jī)制,本文學(xué)習(xí)“Android開發(fā)藝術(shù)探索”基礎(chǔ)上所感,之前也看過網(wǎng)上一些博客講view事件分發(fā)的,看完后仍然有點(diǎn)懵懵懂懂的,在這里謝謝@任玉剛的“Android開發(fā)藝術(shù)探索”,讓我對(duì)view事件分發(fā)有了更深的理解,由于本人技術(shù)有限,希望大家提意見和指正。

一、在介紹點(diǎn)擊事件傳遞規(guī)則之前,首先大家應(yīng)該明白我們研究的對(duì)象是MotionEvent(點(diǎn)擊事件),點(diǎn)擊事件分發(fā)有三個(gè)重要的方法:

public boolean dispatchTouchEvent(MotionEvent ev) 事件分發(fā),如果事件能夠傳遞到當(dāng)前view,此方法一定調(diào)用,返回結(jié)果受當(dāng)前view的onTouchEvent和下級(jí)view的dispatchTouchEvent方法影響,詳細(xì)分析如下:

   1.若當(dāng)前view的onTouchEvent調(diào)用,說明當(dāng)前view調(diào)用onInterceptTouchEvent攔截了事件,同時(shí)當(dāng)前view消耗事件,導(dǎo)致不調(diào)用下級(jí)view的 dispatchTouchEvent,當(dāng)前view的dispatchTouchEvent返回false。
  2.反之,當(dāng)前view的onTouchEvent不調(diào)用調(diào)用,說明當(dāng)前view不調(diào)用onInterceptTouchEvent攔截了事件,同時(shí)當(dāng)前view不消耗事件,導(dǎo)致調(diào)用下級(jí)view的 dispatchTouchEvent將事件分發(fā)下去,當(dāng)前view的dispatchTouchEvent返回true。

public boolean onInterceptTouchEvent(MotionEvent ev) 事件攔截
用來判斷是否攔截某個(gè)事件,如果當(dāng)前view攔截了某個(gè)事件,則在同一事件序列當(dāng)中,此方法不會(huì)再次調(diào)用,返回結(jié)果表示是否攔截事件
public boolean onTouchEvent(MotionEvent ev) 點(diǎn)擊事件處理(消耗事件)

二、三個(gè)方法有什么區(qū)別和聯(lián)系,下面通過偽代碼表示:
public boolean dispatchTouchEvent(MotionEvent ev){ boolean consume=false; if(onInterceptTouchEvent(ev)){ consume=onTouchEvent(ev); }else{ consume=child.dispatchTouchEvent(ev); } return consume; }

    上述偽代碼將三者的關(guān)系表現(xiàn)的淋漓盡致:對(duì)于一個(gè)根viewGroup來說,點(diǎn)擊事件發(fā)生后,首先會(huì)傳遞給當(dāng)前viewGroup,這時(shí)它會(huì)調(diào)用它的dispatchTouchEvent,如果viewGroup的onInterceptTouchEvent返true,表示它攔截當(dāng)前事件,調(diào)用它的onTouchEvent消耗此事件;如果viewGroup的onInterceptTouchEvent返回false就表示它不攔截,當(dāng)前事件會(huì)繼續(xù)傳遞給它的子view,子view調(diào)用dispatchTouchEvent,如此循環(huán)下去直到事件消耗。

當(dāng)一個(gè)view需要處理事件時(shí):
如果它設(shè)置OnTouchListener,那么OnTouchListener中的OnTouch方法一定會(huì)調(diào)用,如果OnTouchListener中 的OnTouch方法返回false,則當(dāng)前view的onTouchEvent方法會(huì)被調(diào)用,反之返回true,則當(dāng)前view的onTouchEvent方法不會(huì)被調(diào)用;
如果當(dāng)前設(shè)置有OnClickListener那么它的OnClick方法會(huì)在onTouchEvent的OnTouch方法之后執(zhí)行,平時(shí)我們常用的OnClickListener,其優(yōu)先級(jí)最低,即處于事件傳遞尾端。
由此可見:在設(shè)置OnTouchListener,OnClickListener且OnTouchListener中 的OnTouch方法返回false前提下,他們執(zhí)行的優(yōu)先級(jí)OnTouchListener >onTouchEvent >OnClickListener(其中的方法)。
關(guān)于事件傳遞的機(jī)制,給出的一些結(jié)論:

1.同一事件序列是指從手指觸屏幕那一刻到手指離開屏幕那一刻,這個(gè)事件序列包含一個(gè)down事件開始,中間包含多個(gè)Move事件, 一個(gè)up事件結(jié)束
2.一個(gè)事件序列只能被一個(gè)view攔截且消耗
3.某個(gè)view一旦決定攔截,那么這一個(gè)事件序列都只能由它來處理(如果時(shí)間序列能夠傳遞給它)并且它的onInterceptTouchEvent不會(huì)再調(diào)用
4.某個(gè)view一旦開始處理事件,如果它不消耗ACTION_DOWN事件(onTouchEvent返回false)那么同一事件序列中其他事件都不會(huì)交個(gè)它處理,并且事件將重新交由它的父元素去處理,即父元素的onTouchEvent 會(huì)被調(diào)用,意思是事件一旦交給一個(gè)view處理,那么它必須消耗掉,否則同一事件序列中剩下的事件就不再交給它來處理,就好比BOSS交給你一個(gè)任務(wù),如果這個(gè)任務(wù)沒有處理好,那么短期內(nèi)BOSS就不敢再把任務(wù)交個(gè)你做了。
5.如果view不消耗ACTION_DOWN以外的其他事件,那么這個(gè)點(diǎn)擊事件會(huì)消失,此時(shí)父元素的onTouchEvent并不會(huì)調(diào)用,并且當(dāng)前view可以持續(xù)收到后續(xù)的事件,最終這些消失的點(diǎn)擊事件會(huì)傳遞給Activity處理,舉個(gè)例子,就好比BOSS交個(gè)你一個(gè)任務(wù),你解決不了,最后還是要交個(gè)BOSS解決一樣的道理。
6.viewGroup默認(rèn)不攔截任何事件。Android源碼中viewGroup的onInterceptTouchEvent方法默認(rèn)返回false。
7.view沒有onInterceptTouchEvent方法,一旦有點(diǎn)擊事件傳遞給它,那么它的onTouchEvent方法就會(huì)調(diào)用。
8.view的onTouchEvent默認(rèn)都可以消耗事件(返回true),除非它是不可點(diǎn)擊的(clickable和longClickable同時(shí)為false),view的longClickable屬性默認(rèn)都是false,clickable屬性要分情況,比如button的clickable默認(rèn)為true,textview的clickable默認(rèn)為false。
9.view的enable屬性不影響onTouchEvent的默認(rèn)返回值。哪怕一個(gè)view是disable狀態(tài)的,只要它的clickable或者longClickable有一個(gè)為true,那么它的onTouchEvent就返回true。
10.事件傳遞過程是由外而內(nèi)的,即事件總是先傳給父元素,然后父元素分發(fā)給子view,但可以通過requestDisallowInterceptTouchEvent方法在子元素中干預(yù)父元素的事件分發(fā),但是ACTION_DOWN除外。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,362評(píng)論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,013評(píng)論 3 423
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 177,346評(píng)論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,421評(píng)論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,146評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,534評(píng)論 1 325
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,585評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,767評(píng)論 0 289
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,318評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,074評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,258評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,828評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,486評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,916評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,156評(píng)論 1 290
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,993評(píng)論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,234評(píng)論 2 375

推薦閱讀更多精彩內(nèi)容