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 事件全部没有了。事实上,也就这种方法也就是这种方法起的作用,

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

posted @ 2017-08-10 12:49  yfceshi  阅读(152)  评论(0编辑  收藏  举报