android系统中的每个ViewGroup的子类都具有下面三个和TouchEvent处理密切相关的方法:

1)public boolean dispatchTouchEvent(MotionEvent ev)          这个方法用来分发TouchEvent

2)public boolean onInterceptTouchEvent(MotionEvent ev)         这个方法用来拦截TouchEvent

3)public boolean onTouchEvent(MotionEvent ev)                 这个方法用来处理TouchEvent

注意:不是所有的View的子类,很多教程都说的是所有的View的子类,只有可以向里面添加View的控件才需要分发,比如TextView它本身就是最小的view了,所以不用再向它的子视图分发了,它也没有子视图了,所以它没有dispatch和Intercept,只有touchEvent。


onInterceptTouchEvent(MotionEvent ev)方法是GroupView控件中特有的一个方法,我们可以实现这个方法来截获所有的窗口点击事件来根据事件的不同来对子控件或自身分发事件触发。覆盖这个方法时需要小心注意,因为它和View.onTouchEvent(MotionEvent)有着复杂的关系,我们使用是需要和View.onTouchEvent(MotionEvent)一同实现来达到正确的结果,事件触发顺序如下:

  1.你首先会在该方法中得到一个down事件。

  2.这个down事件会被得到处理,要么在这个GroupView中子控件的onTouchEvent()方法中,要么在GroupView自身的onTouchEvent()方法中。当我们在onTouchEvent()方法返回值中返回true的话,将会继续看到后续的触发事件(move、up等),要注意的是当我们在GroupView的onTouchEvent()事件返回值中返回false的话,该方法将不再捕获后续的(move up)事件。

  3.如果在该方法返回值中返回false,down事件将根据控件树结构从根节点向目标子控件的onTouchEvent()方法分发。

  4.如果在该方法返回值中返回true,那么你的子控件将获取不到任何点击事件。

 

  为证明上面所说的顺序特点,下面通过测试代码来观察观察:

  


android 并发面试题_截获



1 public class MainActivity extends Activity {
 2     Group1 group1;
 3     Group2 group2;
 4     MyTextView myTv;
 5 
 6     /** Called when the activity is first created. */
 7     @Override
 8     public void onCreate(Bundle savedInstanceState) {
 9         super.onCreate(savedInstanceState);
10         
11         
12         //--group1
13         //----|
14         //-------group2
15         //---------|
16         //------------myTv
17         
18         group1 = new Group1(this);
19         group2 = new Group2(this);
20         myTv = new MyTextView(this);
21         group2.addView(myTv, new LayoutParams(LayoutParams.FILL_PARENT,
22                 LayoutParams.FILL_PARENT));
23         group1.addView(group2, new LayoutParams(LayoutParams.FILL_PARENT,
24                 LayoutParams.FILL_PARENT));
25         setContentView(group1);
26     }
27 }



android 并发面试题_截获


  分别重写Group1和Group2的onInterceptTouchEvent和onTouchEvent方法,重写MyTextView的onTouchEvent方法,最终得到的控件层次结构如下:  

   

android 并发面试题_截获_03

  1.在默认返回值情况下logcat输出如下:  

     

android 并发面试题_android 并发面试题_04

  测试后可知默认情况下和所有方法返回值为false的结果一致,down事件的捕获顺序onInterceptTouchEvent先于onTouchEvent,由于onTouchEvent返回值为false,down事件没被消化,后续的move和up事件没有出现,同时逆序返回到父控件的onTouchEvent方法来捕获,如下图所示:  

   

android 并发面试题_截获_05

 

  2.所有onTouchEvent返回值为true情况下logcat输出如下:

   

android 并发面试题_ontouchevent_06

  输出结果可以看出子控件MyTextView消化了down事件,后续的move和up事件正常捕获,由于down事件被消化,上层的onTouchEvent方法不执行,如下图所示:(三箭头分别指down、move、up事件)

   

android 并发面试题_ontouchevent_07

  既然如此,如果MyTextView中onTouchEvent方法返回为false,而group1和group2的onTouchEvent方法返回true的结果自然也就如下图的顺序了:  

  

android 并发面试题_ontouchevent_08

  测试输出结果证明了这一猜测顺序,  

     

android 并发面试题_截获ontouchevent_09

 

  3.当某个GroupView中的onInterceptTouchEvent方法返回值为true情况下logcat输出如下(如group2):

    

android 并发面试题_ontouchevent_10

  如果在该方法返回值中返回true,那么子控件将获取不到任何点击事件,转而向自身的onTouchEvent方法转发,如下图所示:

  

android 并发面试题_分发事件_11

  如果onTouchEvent方法返回值都为true,那么根据规律结果就如下图顺序触发:  

    

android 并发面试题_截获ontouchevent_12

  最后logcat的结果证实了这一猜测,  

     

android 并发面试题_android 并发面试题_13

  

  根据这一顺序规律我们便可复写GroupView中的onInterceptTouchEvent来控制事件的响应者。