android中Touch事件的處理邏輯

          android中的事件類型分為按鍵事件和屏幕觸摸事件,Touch事件是屏幕觸摸事件的基礎事件,有必要對它進行深入的了解。
          一個最簡單的屏幕觸摸動作觸發了一系列Touch事件:ACTION_DOWN->ACTION_MOVE->ACTION_MOVE->ACTION_MOVE...->ACTION_MOVE->ACTION_UP
          當屏幕中包含一個ViewGroup,而這個ViewGroup又包含一個子view,這個時候android系統如何處理Touch事件呢?到底是ViewGroup來處理Touch事件,還是子view來處理Touch事件呢?我只能很肯定的對你說不一定。呵呵,為什么呢?看看下面我的調查結果你就明白了。
          android系統中的每個View的子類都具有下面三個和TouchEvent處理密切相關的方法:
          1)public boolean dispatchTouchEvent(MotionEvent ev)  這個方法用來分發TouchEvent
          2)public boolean onInterceptTouchEvent(MotionEvent ev) 這個方法用來攔截TouchEvent
          3)public boolean onTouchEvent(MotionEvent ev) 這個方法用來處理TouchEvent
          當TouchEvent發生時,首先Activity將TouchEvent傳遞給最頂層的View, TouchEvent最先到達最頂層 view 的 dispatchTouchEvent ,然后由  dispatchTouchEvent 方法進行分發,如果dispatchTouchEvent返回true ,則交給這個view的onTouchEvent處理,如果dispatchTouchEvent返回 false ,則交給這個 view 的 interceptTouchEvent 方法來決定是否要攔截這個事件,如果 interceptTouchEvent 返回 true ,也就是攔截掉了,則交給它的 onTouchEvent 來處理,如果 interceptTouchEvent 返回 false ,那么就傳遞給子 view ,由子 view 的 dispatchTouchEvent 再來開始這個事件的分發。如果事件傳遞到某一層的子 view 的 onTouchEvent 上了,這個方法返回了 false ,那么這個事件會從這個 view 往上傳遞,都是 onTouchEvent 來接收。而如果傳遞到最上面的 onTouchEvent 也返回 false 的話,這個事件就會“消失”,而且接收不到下一次事件。
          通過語言描述這個處理邏輯很抽象,下面我就用代碼來具體說明一下。
          layout配置文件 main.xml
          <?xml version="1.0" encoding="utf-8"?>
          <test.lzqdiy.MyLinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation
          ="vertical"
              android:layout_width
          ="fill_parent"
              android:layout_height
          ="fill_parent"
              android:gravity
          ="center" >
                 
          <test.lzqdiy.MyTextView 
                      android:layout_width
          ="200px"
                      android:layout_height
          ="200px"
                      android:id
          ="@+id/tv"
                      android:text
          ="lzqdiy"
                      android:textSize
          ="40sp"
                      android:textStyle
          ="bold"
                      android:background
          ="#FFFFFF"
                      android:textColor
          ="#0000FF"/>
          </test.lzqdiy.MyLinearLayout>
          節點層次很簡單,一個LinearLayout中添加了一個TextView。
          下面是java代碼:
          package test.lzqdiy;

          import android.app.Activity;
          import android.os.Bundle;

          public class TestTouchEventApp extends Activity {
              
          /** Called when the activity is first created. */
              @Override
              
          public void onCreate(Bundle savedInstanceState) {
                  
          super.onCreate(savedInstanceState);
                  setContentView(R.layout.main);
              }
          }
          package test.lzqdiy;

          import android.content.Context;
          import android.util.AttributeSet;
          import android.util.Log;
          import android.view.MotionEvent;
          import android.widget.LinearLayout;

          public class MyLinearLayout extends LinearLayout {
              
          private final String TAG = "MyLinearLayout";

              
          public MyLinearLayout(Context context, AttributeSet attrs) {

                  
          super(context, attrs);

                  Log.d(TAG, TAG);

              }

              @Override
              
          public boolean dispatchTouchEvent(MotionEvent ev) {
                  
          int action = ev.getAction();

                  
          switch (action) {

                  
          case MotionEvent.ACTION_DOWN:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_DOWN");

                      
          break;

                  
          case MotionEvent.ACTION_MOVE:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_MOVE");

                      
          break;

                  
          case MotionEvent.ACTION_UP:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_UP");

                      
          break;

                  
          case MotionEvent.ACTION_CANCEL:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_CANCEL");

                      
          break;

                  }
                  
          return super.dispatchTouchEvent(ev);
              }

              @Override
              
          public boolean onInterceptTouchEvent(MotionEvent ev) {

                  
          int action = ev.getAction();

                  
          switch (action) {

                  
          case MotionEvent.ACTION_DOWN:

                      Log.d(TAG, 
          "onInterceptTouchEvent action:ACTION_DOWN");

                      
          break;

                  
          case MotionEvent.ACTION_MOVE:

                      Log.d(TAG, 
          "onInterceptTouchEvent action:ACTION_MOVE");

                      
          break;

                  
          case MotionEvent.ACTION_UP:

                      Log.d(TAG, 
          "onInterceptTouchEvent action:ACTION_UP");

                      
          break;

                  
          case MotionEvent.ACTION_CANCEL:

                      Log.d(TAG, 
          "onInterceptTouchEvent action:ACTION_CANCEL");

                      
          break;

                  }

                  
          return false;

              }

              @Override
              
          public boolean onTouchEvent(MotionEvent ev) {

                  
          int action = ev.getAction();

                  
          switch (action) {

                  
          case MotionEvent.ACTION_DOWN:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_DOWN");

                      
          break;

                  
          case MotionEvent.ACTION_MOVE:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_MOVE");

                      
          break;

                  
          case MotionEvent.ACTION_UP:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_UP");

                      
          break;

                  
          case MotionEvent.ACTION_CANCEL:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_CANCEL");

                      
          break;

                  }

                  
          return true;
              }

          }
          package test.lzqdiy;

          import android.content.Context;
          import android.util.AttributeSet;
          import android.util.Log;
          import android.view.MotionEvent;
          import android.widget.TextView;

          public class MyTextView extends TextView {

              
          private final String TAG = "MyTextView";

              
          public MyTextView(Context context, AttributeSet attrs) {

                  
          super(context, attrs);

              }

              @Override
              
          public boolean dispatchTouchEvent(MotionEvent ev) {
                  
          int action = ev.getAction();

                  
          switch (action) {

                  
          case MotionEvent.ACTION_DOWN:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_DOWN");

                      
          break;

                  
          case MotionEvent.ACTION_MOVE:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_MOVE");

                      
          break;

                  
          case MotionEvent.ACTION_UP:

                      Log.d(TAG, 
          "dispatchTouchEvent action:ACTION_UP");

                      
          break;

                  
          case MotionEvent.ACTION_CANCEL:

                      Log.d(TAG, 
          "onTouchEvent action:ACTION_CANCEL");

                      
          break;

                  }
                  
          return super.dispatchTouchEvent(ev);
              }

              @Override
              
          public boolean onTouchEvent(MotionEvent ev) {

                  
          int action = ev.getAction();

                  
          switch (action) {

                  
          case MotionEvent.ACTION_DOWN:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_DOWN");

                      
          break;

                  
          case MotionEvent.ACTION_MOVE:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_MOVE");

                      
          break;

                  
          case MotionEvent.ACTION_UP:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_UP");

                      
          break;

                  
          case MotionEvent.ACTION_CANCEL:

                      Log.d(TAG, 
          "---onTouchEvent action:ACTION_CANCEL");

                      
          break;

                  }

                  
          return true;

              }

          }
          為了指代方便,下面將MyLinearLayout簡稱為L,將MyTextView簡稱為T,L.onInterceptTouchEvent=true 表示的含義為MyLinearLayout中的onInterceptTouchEvent方法返回值為true,通過程序運行時輸出的Log來說明調用時序。
          第1種情況 L.onInterceptTouchEvent=false&& L.onTouchEvent=true &&T.onTouchEvent=true 輸出下面的Log
          D/MyLinearLayout(11865): dispatchTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(11865): onInterceptTouchEvent action:ACTION_DOWN
          D/MyTextView(11865): dispatchTouchEvent action:ACTION_DOWN
          D/MyTextView(11865): ---onTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(11865): dispatchTouchEvent action:ACTION_MOVE
          D/MyLinearLayout(11865): onInterceptTouchEvent action:ACTION_MOVE
          D/MyTextView(11865): dispatchTouchEvent action:ACTION_MOVE
          D/MyTextView(11865): ---onTouchEvent action:ACTION_MOVE
          ...........省略其他的ACTION_MOVE事件Log
          D/MyLinearLayout(11865): dispatchTouchEvent action:ACTION_UP
          D/MyLinearLayout(11865): onInterceptTouchEvent action:ACTION_UP
          D/MyTextView(11865): dispatchTouchEvent action:ACTION_UP
          D/MyTextView(11865): ---onTouchEvent action:ACTION_UP
          結論:TouchEvent完全由TextView處理。
          第2種情況  L.onInterceptTouchEvent=false&& L.onTouchEvent=true &&T.onTouchEvent=false 輸出下面的Log
          D/MyLinearLayout(13101): dispatchTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13101): onInterceptTouchEvent action:ACTION_DOWN
          D/MyTextView(13101): dispatchTouchEvent action:ACTION_DOWN
          D/MyTextView(13101): ---onTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13101): ---onTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13101): dispatchTouchEvent action:ACTION_MOVE
          D/MyLinearLayout(13101): ---onTouchEvent action:ACTION_MOVE
          ...........省略其他的ACTION_MOVE事件Log
          D/MyLinearLayout(13101): dispatchTouchEvent action:ACTION_UP
          D/MyLinearLayout(13101): ---onTouchEvent action:ACTION_UP
          結論:TextView只處理了ACTION_DOWN事件,LinearLayout處理了所有的TouchEvent。
          第3種情況  L.onInterceptTouchEvent=true&& L.onTouchEvent=true 輸出下面的Log
          D/MyLinearLayout(13334): dispatchTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13334): onInterceptTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13334): ---onTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13334): dispatchTouchEvent action:ACTION_MOVE
          D/MyLinearLayout(13334): ---onTouchEvent action:ACTION_MOVE
          ...........省略其他的ACTION_MOVE事件Log
          D/MyLinearLayout(13334): dispatchTouchEvent action:ACTION_UP
          D/MyLinearLayout(13334): ---onTouchEvent action:ACTION_UP
          結論:LinearLayout處理了所有的TouchEvent。
          第4種情況  L.onInterceptTouchEvent=true&& L.onTouchEvent=false 輸出下面的Log
          D/MyLinearLayout(13452): dispatchTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13452): onInterceptTouchEvent action:ACTION_DOWN
          D/MyLinearLayout(13452): ---onTouchEvent action:ACTION_DOWN
          結論:LinearLayout只處理了ACTION_DOWN事件,那么其他的TouchEvent被誰處理了呢?答案是LinearLayout最外層的Activity處理了TouchEvent。
          好累啊!終于完成了,寫得比較倉促,難免有不妥之處,請博友們指正。





          posted on 2011-05-08 21:36 我為J狂 閱讀(12147) 評論(7)  編輯  收藏 所屬分類: android

          評論

          # re: android中Touch事件的處理邏輯[未登錄] 2011-06-28 17:04 匿名

          沒明白,寫的不好  回復  更多評論   

          # re: android中Touch事件的處理邏輯 2011-07-05 17:03 李倓

          還可以,最后的log算是清晰了  回復  更多評論   

          # re: android中Touch事件的處理邏輯 2011-07-15 14:06 趙林

          寫的很好,很清晰,贊  回復  更多評論   

          # re: android中Touch事件的處理邏輯[未登錄] 2011-12-22 09:34 匿名

          很好 非常感謝  回復  更多評論   

          # re: android中Touch事件的處理邏輯[未登錄] 2012-10-08 17:15 ddd

          一下 最頂層 一下 最外層 你敢在犀利點嗎  回復  更多評論   

          # re: android中Touch事件的處理邏輯 2012-10-09 18:50 LiYa

          你好,我看了下,發現:public boolean onInterceptTouchEvent(MotionEvent ev) 只有在ViewGroup中有,View類本身并沒有該方法喎~~。  回復  更多評論   

          # re: android中Touch事件的處理邏輯 2013-03-13 23:53 啊沖

          很有幫助,謝謝樓主的細心分析  回復  更多評論   


          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          <2011年5月>
          24252627282930
          1234567
          891011121314
          15161718192021
          22232425262728
          2930311234

          導航

          統計

          常用鏈接

          留言簿(11)

          隨筆分類(48)

          文章分類(29)

          常去逛逛

          搜索

          積分與排名

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 东源县| 民丰县| 乌苏市| 天全县| 九龙坡区| 福州市| 龙海市| 若尔盖县| 措美县| 达孜县| 宝鸡市| 平安县| 大冶市| 通渭县| 卫辉市| 宝山区| 澜沧| 南城县| 吴桥县| 合川市| 鹤岗市| 富宁县| 土默特左旗| 黎城县| 黄梅县| 宜兰县| 光山县| 石林| 瓦房店市| 无棣县| 电白县| 达州市| 渭源县| 榆林市| 尚义县| 石柱| 嘉黎县| 兴安县| 高州市| 黎平县| 莱芜市|