关于Android的事件分发机制有很多文章分析的很彻底,比如:
Carson_Ho:https://www.jianshu.com/p/38015afcdb58
Flyzend:https://www.jianshu.com/p/238d1b753e64
Android开发艺术探索
本文是在此基础上对Android的事件分发体系进行总结以应对面试。
1.Touch事件序列?
Touch事件是事件分发的对象,Touch事件序列是指手指从接触屏幕到离开屏幕的过程中产生的一系列事件,包括:MotionEvent.ACTION_DOWN,MotionEvent.ACTION_UP
MotionEvent.ACTION_MOVE,MotionEvent.ACTION_CANCEL等四种事件。
2.事件分发的顺序?
Activity—>ViewGroup—>View
3.事件分发机制的三个重要方法:
dispatchTouchEvent():分发事件 返回值代表是否消费当前事件
onTouchEvent():处理事件 返回值代表是否消费当前事件
onInterceptTouchEvent():拦截事件 返回值代表是否拦截当前事件
4.Activity的事件分发机制?
当一个点击事件产生时。首先会调用Activity的dispatchTouchEvent()方法:
public boolean dispatchTouchEvent(MotionEvent ev) {
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
onUserInteraction();
}
//如果Widow的superDispatchTouchEvent()方法返回true那么事件被Activity拦截掉否则返回Activity的onTouchEvent()的返回值
if (getWindow().superDispatchTouchEvent(ev)) {
return true;
}
return onTouchEvent(ev);
}
getWindow().superDispatchTouchEvent(ev) 方法的返回值由mDecor.superDispatchTouchEvent(event)决定,mDecor本质上是一个FrameLayout即ViewGroup,所以最终调用的是ViewGroup的 dispatchTouchEvent()方法,此方法的返回值为true即Activity的dispatchTouchEvent()的方法的返回值为true
/**
* 说明:
* a. getWindow() = 获取Window类的对象
* b. Window类是抽象类,其唯一实现类 = PhoneWindow类;即此处的Window类对象 =
* PhoneWindow类对象
* c. Window类的superDispatchTouchEvent() = 1个抽象方法,由子类PhoneWindow类实现
*/
@Override
public boolean superDispatchTouchEvent(MotionEvent event) {
return mDecor.superDispatchTouchEvent(event);
// mDecor = 顶层View(DecorView)的实例对象
}
/**
* 说明:
* a. DecorView类是PhoneWindow类的一个内部类
* b. DecorView继承自FrameLayout,是所有界面的父类
* c. FrameLayout是ViewGroup的子类,故DecorView的间接父类 = ViewGroup
*/
public boolean superDispatchTouchEvent(MotionEvent event) {
return super.dispatchTouchEvent(event);
// 调用父类的方法 = ViewGroup的dispatchTouchEvent()
// 即 将事件传递到ViewGroup去处理
}
图解
5.ViewGroup的事件分发机制
图解
图片来源:https://www.jianshu.com/p/38015afcdb58
说明:
当ViewGroup的dispatchTouchEvent()方法调用时,首先会判断onInterceptTouchEvent()方法的返回值,返回true表示拦截事件,则此事件ViewGroup消费,不再向子View传递,而是调用自身的onTouchEvent()方法。返回false,则事件向子View传递,遍历ViewGroup中的所有子View找到点击的View然后调用此子View的dispatchTouchEvent()方法。
6.View的事件分发机制?
图解:
图片来源:https://www.jianshu.com/p/38015afcdb58
public boolean dispatchTouchEvent(MotionEvent event) {
if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&
mOnTouchListener.onTouch(this, event)) {
return true;
}
//以上三个条件都成立时返回true
//1.mOnTouchListener != null 只要我们 mOnTouchListener 在View.setOnTouchListener()中赋值,只要注册Touch,mOnTouchListener就不为空
//2.(mViewFlags & ENABLED_MASK) == ENABLED 用来判断此view是否可用 恒定为true
//3. 手动设置onTouch的返回值
/**button.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
return false;
}
});
若在onTouch()返回true,就会让上述三个条件全部成立,从而使得View.dispatchTouchEvent()直接返回true,事件分发结束
若在onTouch()返回false,就会使得上述三个条件不全部成立,从而使得View.dispatchTouchEvent()中跳出If,执行onTouchEvent(event)
**/
return onTouchEvent(event);
}
说明:当事件到达View层时,View的dispatchTouchEvent()会被调用,然后调用onTouch方法,onTouch()返回true则不再调用onTouchEvent()方法,当然onClick()方法也不在调用,事件分发结束。onTouch()返回false,则dispatchTouchEvent()方法的返回值由onTouchEvent()方法的返回值决定,此时若设置了点击事件则onClick()方法会被调用,事件分发结束。
7.事件分发机制总结:
Android的事件分发机制是从Activity到ViewGroup到View:
一个事件到达后,先调用Activity的dispatchTouchEvent()方法,然后调用Window的dispatchTouchEvent()方法,之后调用DecorView的dispatchTouchEvent()方法,而DecorView本质上是一个ViewGroup,那么此时事件到达ViewGroup,即ViewGroup的dispatchTouchEvent()方法被调用,然后ViewGroup的onInterceptTouchEvent()方法被调用,返回false不拦截,则遍历ViewGroup中的子view,找到点击的子View,此时事件传递到View,view的dispatchTouchEvent()方法被调用,然后view的onTouch()方法会被调用,onTouch()方法返回false(手动复写view的onTouch()方法可改变onTouch的返回值),则会调用view的onTouchEvent()方法,若onTouch()方法返回true,则代表view消费此事件,事件传递结束,若onTouch()方法返回false,则代表view不消费此事件,那么ViewGroup和Activty的onTouch()方法会被依次调用。