Android 事件分发机制

在Android中事件分发是很重要的一块知识,了解并熟悉整套的分发机制有助于更好的分析各种点击滑动失效问题,更好去扩展控件的事件功能和开发自定义控件,同时事件分发机制也是Android面试必问考点之一。

MotionEvent事件初探

MotionEvent在Android View事件分发、处理过程中很重要的一个类,我们对屏幕的点击,滑动,抬起等一系的动作都是由一个一个MotionEvent对象组成的。根据不同动作,主要有以下三种事件类型:

  • ACTION_DOWN:手指刚接触屏幕,按下去的那一瞬间产生该事件
  • ACTION_MOVE:手指在屏幕上移动时候产生该事件
  • ACTION_UP:手指从屏幕上松开的瞬间产生该事件

从ACTION_DOWN开始到ACTION_UP结束我们称为一个事件序列
正常情况下,无论你手指在屏幕上有多么骚的操作,最终呈现在MotionEvent上来讲无外乎下面两种:

  • 点击后抬起,也就是单击操作:ACTION_DOWN -> ACTION_UP
  • 点击后再风骚的滑动一段距离,再抬起:ACTION_DOWN -> ACTION_MOVE -> ... -> ACTION_MOVE -> ACTION_UP

事件分发中重要方法

事件分发需要View的三个重要方法来共同完成:

  • public boolean dispatchTouchEvent(MotionEvent event)
    如果一个MotionEvent传递给了View,那么dispatchTouchEvent方法一定会被调用!
    返回值:表示是否消费了当前事件。可能是View本身的onTouchEvent方法消费,也可能是子View的dispatchTouchEvent方法中消费。
    返回 true表示事件被消费,本次的事件终止传递。
    返回 false表示View以及子View均没有消费事件,将调用父View的onTouchEvent方法。

  • **public boolean onInterceptTouchEvent(MotionEvent ev) **
    事件拦截,当一个ViewGroup在接到MotionEvent事件序列时候,首先会调用此方法判断是否需要拦截。
    特别注意,这是ViewGroup特有的方法,View并没有拦截方法
    返回值:是否拦截事件传递,返回true表示拦截了事件,那么事件将不再向下分发而是调用View本身的onTouchEvent方法。返回false表示不做拦截,事件将向下分发到子View的dispatchTouchEvent方法。

  • public boolean onTouchEvent(MotionEvent ev)
    真正对MotionEvent进行处理或者说消费的方法。在dispatchTouchEvent进行调用。
    返回值:返回true表示事件被消费,本次的事件终止。返回false表示事件没有被消费,将调用父View的onTouchEvent方法

以下是伪代码(基于ViewGroup源码)描述以上三个方法的关系:

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
            // 检查是否拦截该事件 
            final boolean intercepted;
            if (actionMasked == MotionEvent.ACTION_DOWN
                    || mFirstTouchTarget != null) {
                final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;
                // 子View请求是否需要父View拦截标志
               if (!disallowIntercept) {
                    // 调用onIntercepteTouchEvent判断时候拦截该事件
                    intercepted = onInterceptTouchEvent(ev);
                } 
            } 
        
        if (intercepted)  {
           // 该事件被拦截,交由onTouchEvent方法来处理
           handled = onTouchEvent(ev)
        } else {
           // 该事件未被拦截,交由其子类的dispatchTouchEvent处理
           handled = child.dispatchTouchEvent(ev);
        }
        return handled;
    }

我们再看看View的dispatchTouchEvent方法是如何实现的:

    public boolean dispatchTouchEvent(MotionEvent event) {
        // 事件是否被消费
        boolean result = false;
        // 相关的监听器信息
        ListenerInfo li = mListenerInfo;
        if (li != null && li.mOnTouchListener != null
                && (mViewFlags & ENABLED_MASK) == ENABLED
                && li.mOnTouchListener.onTouch(this, event)) {
            // 检查事件被OnTouchListener的onTouch方法消费了
            result = true;
        }
        // 如果没有被执行,才执行onTouchEvent
        if (!result && onTouchEvent(event)) {
            result = true;
        }

        return result;
    }

在ViewGroup的dispatchTouchEvent我们发现disallowIntercept这个标志, 通过它我们可以找到另一个重要的方法requestDisallowInterceptTouchEvent:

  • requestDisallowInterceptTouchEvent
    requestDisallowInterceptTouchEvent 是ViewGroup类中的一个公用方法, android系统中,一次点击事件是从父view传递到子view中,每一层的view可以决定是否拦截并处理点击事件或者传递到下一层,如果子view不处理点击事件,则该事件会传递会父view,由父view去决定是否处理该点击事件。但子view可以通过设置此方法去告诉父view不要拦截并处理点击事件,父view应该接受这个请求直到此次点击事件结束。

实际的应用中,可以在子view的onTouch事件中注入父ViewGroup的实例,并调用requestDisallowInterceptTouchEvent去阻止父view拦截点击事件

public boolean onTouchEvent(View v, MotionEvent event) {
     ViewGroup viewGroup = (ViewGroup) v.getParent();
     switch (event.getAction()) {
     case MotionEvent.ACTION_MOVE: 
         viewGroup.requestDisallowInterceptTouchEvent(true);
         break;
     case MotionEvent.ACTION_UP:
     case MotionEvent.ACTION_CANCEL:
         viewGroup .requestDisallowInterceptTouchEvent(false);
         break;
     }
}

事件分发过程

了解Android整个事件分发过程,比较好的方式是写一个Demo,步骤大致是先重写一个ViewGroup,一个View,然后重写Activity的dispatchTouchEvent、onTouchEvent,ViewGroup的dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent以及View的dispatchTouchEvent、onTouchEvent,然后针对各个Event打Log,网上的很多关于Android事件分发的博文都是这么讲解的, 有兴趣可以自行查找。
通过下面的流程图,会更加清晰的帮助我们梳理事件分发机制:

Android事件流程图

<center> View结构图</center >

<center> View事件分发流程图</center >

  • 仔细看的话, 从上往下依次是Activity(Window)、ViewGroup(DecorView, RootViewGroup)、View, Activity是事件的起源。
    *事件由Activity的dispatchTouchEvent做分发, 传递到ViewGroup的dispatchTouchEvent, onIntercepterTouchEvent,接着传递到View的dispatchTouchEvent。
  • 如果事件不被消费,整个事件又会回传到顶层Activity
  • dispatchTouchEvent 和 onTouchEvent 一旦return true,事件就停止传递了(到达终点)。
  • dispatchTouchEvent 和 onTouchEvent return false的时候事件都回传给父控件的onTouchEvent处理。
  • ViewGroup 想把自己分发给自己的onTouchEvent,需要拦截器onInterceptTouchEvent方法return true,而ViewGroup 的拦截器onInterceptTouchEvent 默认是不拦截的。
  • View 没有拦截器,为了让View可以把事件分发给自己的onTouchEvent,View的dispatchTouchEvent默认实现(super)就是把事件分发给自己的onTouchEvent。

OnTouch & OnTouchEvent的关系

其实在前面已经提到过,还是直接看源码

   public boolean dispatchTouchEvent(MotionEvent event) {
        // 事件是否被消费
        boolean result = false;
        // 相关的监听器信息
        ListenerInfo li = mListenerInfo;
        if (li != null && li.mOnTouchListener != null
                && (mViewFlags & ENABLED_MASK) == ENABLED
                && li.mOnTouchListener.onTouch(this, event)) {
            // 检查事件被OnTouchListener的onTouch方法消费了
            result = true;
        }
        // 如果没有被执行,才执行onTouchEvent
        if (!result && onTouchEvent(event)) {
            result = true;
        }

        return result;
    }

onTouch是OnTouchListener的接口方法,它是用来获取Touch 信息用的,onTouchEvent方法用来处理诸如down, move, up的消息, 通过以上源码可以看出:
onTouchListener的onTouch方法优先级比onTouchEvent高,会先触发。
假如onTouch方法返回false会接着触发onTouchEvent,反之onTouchEvent方法不会被调用。
内置诸如click事件的实现等等都基于onTouchEvent,假如onTouch返回true,这些事件将不会被触发。

OnTouchEvent & OnClick & OnLongClick的关系

还是来看下伪代码

 public boolean onTouchEvent(MotionEvent event) {
    switch (action) {
       case MotionEvent.ACTION_UP:
         if (!mHasPerformedLongPress && !mIgnoreNextUpEvent) {
            // This is a tap, so remove the longpress check
            removeLongPressCallback();
            if (!focusTaken) {
                if (mPerformClick == null) {
                    mPerformClick = new PerformClick();
                }
                if (!post(mPerformClick)) {
                    // 触发OnClickListener的onClick回调
                    performClick();
                }
            }
        }
        break:
      case MotionEvent.ACTION_DOWN:
        setPressed(true, x, y);
        checkForLongClick(0, x, y);
        break;
      case MotionEvent.ACTION_CANCEL:
        setPressed(false);
        removeTapCallback();
        removeLongPressCallback();
        break;
    }
 }

以上源码看一看出,onClick是在onTouchEvent ACTION_UP的时候出发的,如果该View的onTouchEvent或者它的ACTION_UP不被调用是不会触发onClick回调的。

关于onLongClick就稍微复杂一些,通过以上源码可以看到,当用户按下时(ACTION_DOWN)执行了checkForLongClick方法

private void checkForLongClick(int delayOffset, float x, float y) {
        if ((mViewFlags & LONG_CLICKABLE) == LONG_CLICKABLE) {
            mHasPerformedLongPress = false;

            if (mPendingCheckForLongPress == null) {
                mPendingCheckForLongPress = new CheckForLongPress();
            }
            mPendingCheckForLongPress.setAnchor(x, y);
            mPendingCheckForLongPress.rememberWindowAttachCount();
            postDelayed(mPendingCheckForLongPress,
                    ViewConfiguration.getLongPressTimeout() - delayOffset);
        }
    }

该方法实际就是创建了个CheckForLongPress 的 Runnable, 它的run方法里面就是performLongClick(OnLongClickListener的回调就是在该方法中执行的),然后把它放入一个HandlerActionQueue队列中,让它延迟一段时间(DEFAULT_LONG_PRESS_TIMEOUT = 500)以后执行。
同时如果500毫秒之内,ACTION_UP或者ACTION_CANCEL执行了,就会移除该Runnable。

参考

图解 Android 事件分发机制
一文读懂Android View事件分发机制
一文解决Android View滑动冲突

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,277评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,689评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,624评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,356评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,402评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,292评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,135评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,992评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,429评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,636评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,785评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,492评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,092评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,723评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,858评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,891评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,713评论 2 354

推荐阅读更多精彩内容