• Android 的事件传递机制,详解


    Android 的事件传递机制,详解

    前两天和一个朋友聊天的时候。然后说到事件传递机制。然后让我说的时候,忽然发现说的不是非常清楚,事实上Android 的事件传递机制也是知道一些,可是感觉自己知道的非常模糊。不过知道事件是从外层先传到内层,在从内存最后回馈到外层,可是详细的几个方法的调用过程。自己却知道的不是非常详细,我想非常多人都是这样的情况,然后自己就上网去查,然后看到的全部都是在讲会调用哪个几个方法。可是基本没有讲这几个方法的详细作用。

    自己回去写了demo 。今天就把自己的理解写出来。希望大家能更详细了解这几个方法的详细作用。

    首先事件传递会用到的几个方法,我们先列举一下。

    1. public boolean onInterceptTouchEvent(MotionEvent ev)
    2. public boolean dispatchTouchEvent(MotionEvent ev)
    3. public boolean onTouchEvent(MotionEvent ev)

    我们查看源代码,能够看到,onInterceptTouchEvent 这种方法是viewGroup 的方法

    可是 dispatchTouchEvent 和 onTouchEvent 都是view 方法。

    那么我们先写一个View OutView(外层view)继承ViewGroup ,MiddleView(中层View) 继承ViewGroup ,InnerView (内层Button,Button 也是继承的view )继承View

    既然 dispatchTouchEvent 和 onTouchEvent 是view 的方法,那么我们在每一个view 里面重写这两个方法。例如以下。

    每一个方法会打印两边log,第一个log,是调用这种方法開始的时候,就打一句,第二个log ,是这种方法调用父类,也就说,该走的逻辑走完,而且有返回值的时候。为了log 清楚。都不过打印了 ACTION_DOWN 事件的。

     @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "dispatchTouchEvent");
                    break;
            }
    
            boolean flag = super.dispatchTouchEvent(ev);
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "dispatchTouchEvent:" + flag);
                    break;
            }
    
    
            return flag;
        }
    
    
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onTouchEvent");
                    break;
            }
    
            boolean flag = super.onTouchEvent(ev);
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onTouchEvent:" + flag);
                    break;
            }
    
    
            return flag;
    
    
        }
    

    输出的log 例如以下

    toucheventdemo E/OutView﹕ dispatchTouchEvent
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent
    toucheventdemo E/InnerView﹕ dispatchTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent:true
    toucheventdemo E/InnerView﹕ dispatchTouchEvent:true
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent:true
    toucheventdemo E/OutView﹕ dispatchTouchEvent:true

    能够看到前三个尽管都是直接都是dispatchTouchEvent 方法里面的log 打印的,可是log并没有返回值,也就是说,都是运行的方法里面第一句log。
    可是当运行到InnerView﹕ dispatchTouchEvent 的时候。又运行了InnerView﹕ onTouchEvent 的方法

    事实上是这样的, boolean flag = super.dispatchTouchEvent; 这一句,super 中,运行的dispatchTouchEvent 。
    事实上会做两件事

    1。第一件事。是看当前的view 有没有子view,假设有子view,而且会推断当前的触摸点是否在子view 上面。假设在的话。那么就会调用子view的dispatchTouchEvent  ,然后一层一层往里面调用,
    2.一直调用到没有子view 的时候。然后停止dispatchTouchEvent ,由于InnerView 是继承的view,当然没有子view,然后这个时候,就開始调用子view 的onTouchEvent。
    
    onTouchEvent 返回 true 。那么当前view 的dispatchTouchEvent  也会返回view, 此时,就会讲dispatchTouchEvent 的分发结果反馈给父view,当前view 的父view 的dispatchTouchEvent 也会返回true,以此类推,返回到最外层。
    

    那么假设里层的view 返回的true 那假设最里层的view onTouchEvent返回的是false 那?我们将里层的view 继承改成View,

    toucheventdemo E/OutView﹕ dispatchTouchEvent
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent
    toucheventdemo E/InnerView﹕ dispatchTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent:false
    toucheventdemo E/InnerView﹕ dispatchTouchEvent:false
    toucheventdemo E/MiddleView﹕ onTouchEvent
    toucheventdemo E/MiddleView﹕ onTouchEvent:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent:false
    toucheventdemo E/OutView﹕ onTouchEvent
    toucheventdemo E/OutView﹕ onTouchEvent:false
    toucheventdemo E/OutView﹕ dispatchTouchEvent:false

    我们会发现,当里层的view ,onTouchEvent:false 的时候。那么里层的dispatchTouchEvent 也是false ,而且此时立刻会调用。里层view 的父view(也就是MiddleView) 的 onTouchEvent 事件,MiddleView的onTouchEvent 也返回false ,那么此时MiddleView dispatchTouchEvent 也会是false,然后接着调用了最外层 OutView onTouchEvent 方法。

    从这两个小试验,你能看出,dispatchTouchEvent 的返回结果是怎么得来的吗?

    分两种情况

    1. 当前 没有子view 那么当前view 的 dispatchTouchEvent 结果就是当前view onTouchEvent 的结果

    2. 当前view 有子view, 此时又分两种情况

      1. 子view 的 dispatchTouchEvent 返回false 那么当前view 的 dispatchTouchEvent 结果參照第一种情况,也就是当前view onTouchEvent 运行 ,而且dispatchTouchEvent 的结果是 当前view 的 onTouchEvent 的结果
      2. 子view 的 onTouchEvent 返回true 。那么 此时当前view 的onTouchEvent 的不运行 直接直接返回的是子view 的 dispatchTouchEvent 返回的view
    将两种情况再总结一下。就是。当前view的 dispatchTouchEvent 的返回结果是:假设子view 的 dispatchTouchEvent 返回true ,那么当前view 不运行onTouchEvent ,当前view 的 dispatchTouchEvent 直接返回true, 假设子view 的dispatchTouchEvent 返回false ,当前view 的 dispatchTouchEvent的结果看 看当前view 的onTouchEvent 结果。

    看到这里,你是否对。 dispatchTouchEvent onTouchEvent 略微有一些理解了那?事实上我们看方法的名称也应该能理解一些,dispatchTouchEvent 翻译:触摸事件分发, onTouchEvent 触摸事件,也就是说dispatchTouchEvent 主要做的是处理的触摸事件的分发管理,用来告诉父view 当前view 有没有处理我们的触摸事件。而且处理,当前触摸的点,还有没有子view, onTouchEvent 用来处理我们详细的触摸事件。

    (详细这里为什么说还要推断当前触摸的点还有没有子view。你能够自己试验,触摸MiddleView 之内。InnerView之外的区域。看会不会打印InnerView 里面的log )

    说了半天,我们一直在说dispatchTouchEvent onTouchEvent 这两个方法。那么onInterceptTouchEvent 这种方法又是干嘛的那?

    首先翻译api onInterceptTouchEvent : 拦截TouchEvent 事件。

    应为onInterceptTouchEvent 是viewGroup 的方法,那么我们重写OutView 和 MiddleView 的onInterceptTouchEvent方法

    
        public boolean onInterceptTouchEvent(MotionEvent ev) {
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent");
                    break;
            }
    
            boolean flag = super.onInterceptTouchEvent(ev);
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent:" + flag);
                    break;
            }
    
    
            return flag;
        }
    
    
    然后运行。点击InnerView  看log 
    

    toucheventdemo E/OutView﹕ dispatchTouchEvent
    toucheventdemo E/OutView﹕ onInterceptTouchEvent
    toucheventdemo E/OutView﹕ onInterceptTouchEvent:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent:false
    toucheventdemo E/InnerView﹕ dispatchTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent
    toucheventdemo E/InnerView﹕ onTouchEvent:false
    toucheventdemo E/InnerView﹕ dispatchTouchEvent:false
    toucheventdemo E/MiddleView﹕ onTouchEvent
    toucheventdemo E/MiddleView﹕ onTouchEvent:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent:false
    toucheventdemo E/OutView﹕ onTouchEvent
    toucheventdemo E/OutView﹕ onTouchEvent:false
    toucheventdemo E/OutView﹕ dispatchTouchEvent:false

    我们会发现,在全部的详细运行 dispatchTouchEvent 的方法一開始运行。都会先运行当前view 的onInterceptTouchEvent 结果。onInterceptTouchEvent 的默认返回值是false

    如今我们改动一下MiddleView 的onInterceptTouchEvent 返回值为true 看代码

        public boolean onInterceptTouchEvent(MotionEvent ev) {
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent");
                    break;
            }
    
            boolean flag = super.onInterceptTouchEvent(ev);
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent:" + true);
                    break;
            }
    
    
            return true;
        }

    然后我们运行,看log

    toucheventdemo E/OutView﹕ dispatchTouchEvent
    toucheventdemo E/OutView﹕ onInterceptTouchEvent
    toucheventdemo E/OutView﹕ onInterceptTouchEvent:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent:true
    toucheventdemo E/MiddleView﹕ onTouchEvent
    toucheventdemo E/MiddleView﹕ onTouchEvent:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent:false
    toucheventdemo E/OutView﹕ onTouchEvent
    toucheventdemo E/OutView﹕ onTouchEvent:false
    toucheventdemo E/OutView﹕ dispatchTouchEvent:false
    

    我们会发现,当改动了,MiddleView 的onInterceptTouchEvent 返回结果是true 之后,
    本来应该向InnerView 做分发的那部分log 没有了。然后直接运行了当前MiddleView 的 onTouchEvent 结果,然后运行完
    onTouchEvent结果,之后,当前view 的dispatchTouchEvent 的结果也出来了,然后紧接着把这个有反馈给了OutView 外层view 的。而且运行的原理和我们前面只重写 dispatchTouchEvent onTouchEvent 的原理一样。

    到这里你也应该知道这个onInterceptTouchEvent 是干嘛的吧,他事实上就是拦截事件是否要继续往下传递。假设返回true ,那么代表 事件被当前view 拦截,而且不向下传递,直接运行当前view 的 onTouchEvent 事件。

    到这里,我想应该清楚这几个方法的含义了吧。

    1. onInterceptTouchEvent : 拦截事件。推断是否向下传递。

      返回true 代表拦截。返回false 代表不拦截

    2. dispatchTouchEvent : 事件分发,管理事件的分发,向子view 分发事件,而且,依据得到的子view的事件处理情况,来推断是否响应当前view的 触摸事件 返回true,代表当前view 的或者子view 已经处理事件,告诉父view, 不须要处理事件,返回false ,代表当前view 或者子view 也没有处理事件,告诉父view,能够去处理事件了。

    3. onTouchEvent : 事件的处理,在此方法内处理事件。

      返回true ,代表当前view 处理了事件,false 代表当前view 没有处理事件。

    然后我们再看另外一个可能经常使用的方法。

    public void requestDisallowInterceptTouchEvent(boolean disallowIntercept)

    首先翻译: 请求禁止拦截触摸事件 ,字面理解,就是禁止吊我们的拦截。也就是禁止吊这种方法。onInterceptTouchEvent

    那么我们如今把全部的onInterceptTouchEvent dispatchTouchEvent onTouchEvent 都加上move 事件。

    代码例如以下,

    
        public boolean onInterceptTouchEvent(MotionEvent ev) {
    
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_DOWN");
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_MOVE");
                    break;
            }
    
            boolean flag = super.onInterceptTouchEvent(ev);
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_DOWN:" + flag);
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_MOVE:" + flag);
                    break;
            }
    
    
            return flag;
        }
    
    
        @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "dispatchTouchEvent ACTION_DOWN");
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "dispatchTouchEvent ACTION_MOVE");
                    break;
            }
    
            boolean flag = super.dispatchTouchEvent(ev);
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "dispatchTouchEvent ACTION_DOWN:" + flag);
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "dispatchTouchEvent ACTION_MOVE:" + flag);
                    break;
            }
    
    
            return flag;
        }
    
    
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onTouchEvent ACTION_DOWN");
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onTouchEvent ACTION_MOVE" );
                    break;
            }
    
            boolean flag = super.onTouchEvent(ev);
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onTouchEvent ACTION_DOWN:" + flag);
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onTouchEvent ACTION_MOVE:" + flag);
                    break;
            }
    
    
            return flag;
    
    
        }
    

    而且给 MiddleView 的onInterceptTouchEvent 方法中加上 requestDisallowInterceptTouchEvent(true);
    例如以下

     public boolean onInterceptTouchEvent(MotionEvent ev) {
    
            //加上此句
            requestDisallowInterceptTouchEvent(true);
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_DOWN");
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_MOVE");
                    break;
            }
    
            boolean flag = super.onInterceptTouchEvent(ev);
    
    
            switch (ev.getAction()) {
    
                case MotionEvent.ACTION_DOWN:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_DOWN:" + flag);
                    break;
                case MotionEvent.ACTION_MOVE:
                    Log.e(TAG, "onInterceptTouchEvent ACTION_MOVE:" + flag);
                    break;
            }
    
    
            return flag;
        }

    那么如今我们触摸,然后看log ,可能log 比較多。可是希望大家慢慢细致看。

     toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_DOWN
    toucheventdemo E/OutView﹕ onInterceptTouchEvent ACTION_DOWN
    toucheventdemo E/OutView﹕ onInterceptTouchEvent ACTION_DOWN:false
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_DOWN
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent ACTION_DOWN
    toucheventdemo E/MiddleView﹕ onInterceptTouchEvent ACTION_DOWN:false
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_DOWN
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_DOWN
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_DOWN:true
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_DOWN:true
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_DOWN:true
    toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_DOWN:true
    
    //从这里開始move事件。
    toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_MOVE:true
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_MOVE:true
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_MOVE:true
    toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_MOVE:true
    toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_MOVE
    toucheventdemo E/InnerView﹕ onTouchEvent ACTION_MOVE:true
    toucheventdemo E/InnerView﹕ dispatchTouchEvent ACTION_MOVE:true
    toucheventdemo E/MiddleView﹕ dispatchTouchEvent ACTION_MOVE:true
    toucheventdemo E/OutView﹕ dispatchTouchEvent ACTION_MOVE:true

    我们能够看处理啊,第一次是ACTION_DOWN 事件的时候一切正常,(InnerView 是继承的 Button,所以onTouchEvent 返回的是true)

    可是从ACTION_MOVE 你会发现,全部的onInterceptTouchEvent 事件全部没有了。事实上,也就这种方法也就是这种方法起的作用,

    好了。全部的东西讲完了,不知道你如今是否对事件传递有足够的理解那?

  • 相关阅读:
    洛谷P4315 月下“毛景树”(边权维护)
    Water Tree CodeForces
    P3384 【模板】树链剖分
    Codeforces 161D Distance in Tree(树的点分治)
    git命令及使用方法
    vue下拉刷新,下拉加载更多
    Vue中 v-for 生成样式并默认选中第一个样式
    js面向对象实现分页版轮播图
    js面向对象实现购物车
    js面向对象实现放大镜
  • 原文地址:https://www.cnblogs.com/yfceshi/p/7338688.html
Copyright © 2020-2023  润新知