对于Android事件拦截机制,相信对于大多数Android初学者是一个抓耳挠腮难于理解的问题。其实理解这个问题并不困难。

  首先,你的明白事件拦截机制到底是怎么一回事?这里说的事件拦截机制,指的是对触摸事件的拦截机制。那何为触摸事件?所谓的触摸事件,就是指系统捕获的触摸屏幕所产生的事件。当我们点击按钮时候,此时其实就产生了三个事件。按钮按下,这是事件之一;如果你不小心滑动了一点儿,这是事件之二;如果你抬起,这是事件之三。Android为我们这个触摸事件封装了一个类——MotionEvent。在OnTouchEvent事件中,能够非常方便的监听这三个事件。

  既然,能够监听这个触摸事件,那么何来事件拦截之说了。那么,接下来,请同学们设想这样一个场景好吗?一个View放在一个ViewGroup里面,这个父ViewGroup控件又放在另外一个VIewGroup里面,甚至还可以继续嵌套,这样子子孙孙无穷尽彦。那么问题来了呀?可触摸事件这里只有一个呀,我到底把他给谁。爹爹ViewGroup与子View都想处理这个触摸事件了,于是"事件拦截"这个很霸气的名字就应运而生。

  要理解这个事件拦截机制,我这里需要置身于一个设身处地的场景。就好比你所在一个公司,有一个CEO,CEO下面有总监,总监下面是经理,经理下面有个苦逼的你。此时,来了一项任务以后,CEO把他分配总监,总监分配给经理,经理就把他交给你。这样任务上传下达的流程,就与事件分发与拦截流程蛮像了。

  为了事件更好的理解这个案例,我这里就用控件结构模拟这样的组织结构,来较深入的讲解事件拦截机制。

  我们看一下这个控件UI架构图:

  怎么在判断是父容器与子控件发生了事件拦截,在父容器(ViewGroup)中监听OnTouchEvent()事件,DispatchEvent()事件与OninterceptEvent()事件,在子View监听OnTouchEvent()事件与DispatchEvent()事件。怎么看这些事件进行调用了,我们在每个事件打印相应的日志,就ok了。好好好,有了这样大体一个思路,我们就可以上源代码了。

  子View源代码如下:

public class MyView extends View {

    public MyView(Context context) {
super(context);
} public MyView(Context context, AttributeSet attrs) {
super(context, attrs);
} public MyView(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
} @Override
public boolean onTouchEvent(MotionEvent event) {
Log.d("zcw", "View onTouchEvent" + event.getAction());
return super.onTouchEvent(event);
} @Override
public boolean dispatchTouchEvent(MotionEvent event) {
Log.d("zcw", "View dispatchTouchEvent" + event.getAction());
return super.dispatchTouchEvent(event);
} }

  父ViewGroup源代码如下:

public class MyViewGroupC extends LinearLayout {

    public MyViewGroupC(Context context) {
super(context);
} public MyViewGroupC(Context context, AttributeSet attrs) {
super(context, attrs);
} public MyViewGroupC(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
} @Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupC dispatchTouchEvent" + ev.getAction());
return super.dispatchTouchEvent(ev);
} @Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupC onInterceptTouchEvent" + ev.getAction());
return super.onInterceptTouchEvent(ev);
} @Override
public boolean onTouchEvent(MotionEvent event) {
Log.d("zcw", "ViewGroupC onTouchEvent" + event.getAction());
return super.onTouchEvent(event);
}
}

  祖ViewGroup的源代码如下:

public class MyViewGroupB extends LinearLayout {

    public MyViewGroupB(Context context) {
super(context);
} public MyViewGroupB(Context context, AttributeSet attrs) {
super(context, attrs);
} public MyViewGroupB(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
} @Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupB dispatchTouchEvent" + ev.getAction());
return super.dispatchTouchEvent(ev);
} @Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupB onInterceptTouchEvent" + ev.getAction());
return super.onInterceptTouchEvent(ev);
} @Override
public boolean onTouchEvent(MotionEvent event) {
Log.d("zcw", "ViewGroupB onTouchEvent" + event.getAction());
return super.onTouchEvent(event);
}
}

  曾祖ViewGroup控件的源代码如下:

public class MyViewGroupA extends LinearLayout {

    public MyViewGroupA(Context context) {
super(context);
} public MyViewGroupA(Context context, AttributeSet attrs) {
super(context, attrs);
} public MyViewGroupA(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
} @Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupA dispatchTouchEvent" + ev.getAction());
return super.dispatchTouchEvent(ev);
} @Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d("zcw", "ViewGroupA onInterceptTouchEvent" + ev.getAction());
return super.onInterceptTouchEvent(ev);
} @Override
public boolean onTouchEvent(MotionEvent event) {
Log.d("zcw", "ViewGroupA onTouchEvent" + event.getAction());
return super.onTouchEvent(event);
}
}

  我们在每个要监听事件里面都埋下了点。

  我们运行程序,看每个控件的事件执行的先后顺序:

  我们触摸子View时候,发现越顶层父控件控件DispatchEvent,onInterceptTouchEvent事件越先执行。当里层OnTouchEvent方法执行完全,然后由里到外执行OnTouchEvent方法。执行流程图是这样的:

  我们稍微修改一下子控件的代码,在onTouchEvent方法中返回true的话,此时运行效果就是这样的了。

  我们,就可以得出来这样的结论,如果是return true 以后,就是把该事件进行截取,要不向下传递的事件。这就是android事件拦截的本质。

  return false——事件放任自流,该传递的就进行传递,return true——自己做了,不麻烦别人了,事件截获,不进行传递了。

  这就是我对事件拦截机制的总结,恳请大家吐槽。

Android 事件拦截机制一种粗鄙的解释的更多相关文章

  1. 讲讲Android事件拦截机制

    简介 什么是触摸事件?顾名思义,触摸事件就是捕获触摸屏幕后产生的事件.当点击一个按钮时,通常会产生两个或者三个事件--按钮按下,这是事件一,如果滑动几下,这是事件二,当手抬起,这是事件三.所以在And ...

  2. Android事件拦截机制简单分析

    前一阶段,在学习的时候,遇到了我觉得的我接触安卓以来的最多的一次事件拦截出来,那个项目,用到了slidemenu側滑菜单条,然后加上tab标签,还有轮播广告,listview上下滑动.viewpage ...

  3. Android事件拦截机制 - 两句话

    模拟情形:ViewGroupA ->ViewGroupB->View False往下走,True就停下.(适用于事件传递和事件处理)

  4. android事件拦截处理机制详解

    前段时间刚接触过Android手机开发,对它的事件传播机制不是很了解,虽然网上也查了相关的资料,但是总觉得理解模模糊糊,似是而非,于是自己就写个小demo测试了一下.总算搞明白了它的具体机制.写下自己 ...

  5. Android开发系列之事件拦截机制

    对于Android开发者来说理解事件传递机制的重要性,我想应该是不言而喻的.在一个Activity里面,我们经常会重写onTouchEvent事件,可是重写结束之后,对于是返回true还是返回fals ...

  6. android事件拦截处理机制具体解释

    前段时间刚接触过android手机开发.对它的事件传播机制不是非常了解,尽管网上也查了相关的资料,可是总认为理解模模糊糊,似是而非,于是自己就写个小demo測试了一下. 总算搞明确了它的详细机制.写下 ...

  7. android事件分发机制

    android事件分发机制,给控件设置ontouch监听事件,当ontouch返回true时,他就不会走onTouchEvent方法,要想走onTouchEvent方法只需要返回ontouch返回fa ...

  8. Android事件分发机制完全解析,带你从源码的角度彻底理解

    Android事件构成 在Android中,事件主要包括点按.长按.拖拽.滑动等,点按又包括单击和双击,另外还包括单指操作和多指操作.所有这些都构成了Android中的事件响应.总的来说,所有的事件都 ...

  9. Android事件传递机制(转)

    Android事件构成 在Android中,事件主要包括点按.长按.拖拽.滑动等,点按又包括单击和双击,另外还包括单指操作和多指操作.所有这些都构成了Android中的事件响应.总的来说,所有的事件都 ...

随机推荐

  1. margin 碰到过的重叠问题

    双倍margin 稍微记录一下,因为常忘 float + margin,实际边距会变成设置值的双倍. 解决办法是 将元素设为 display:inline 最好要了解 IE的 haslayout 问题 ...

  2. CSS布局(二)

    本节内容:position.float.clear.浮动布局例子.百分比宽度 position CSS中的position属性设置元素的位置.属性值:static.relative.fixed.abs ...

  3. GPU高性能计算-CUDA

    前段时间有个同学的毕设是搞并行计算的,他基本不懂编程把我拉过去帮忙,我之前也没弄过,帮着搞了2天.先把代码贴上去,等有时间在把详细补充一些内容. CUDA编程主要是利用了显卡优越的并行计算能力,把一个 ...

  4. freeCodeCamp:Where do I belong

    我身在何处? 先给数组排序,然后找到指定的值在数组的位置,最后返回位置对应的索引. 举例:where([1,2,3,4], 1.5) 应该返回 1.因为1.5插入到数组[1,2,3,4]后变成[1,1 ...

  5. R语言读取excel文件的3种方法

    R读取excel文件中数据的方法: 电脑有一个excel文件,原始的文件路径是:E:\R workshop\mydata\biom excel数据为5乘2阶矩阵,元素为                ...

  6. Oracle 设置表空间自增长

    Oracle修改表空间大小 使用Oracle10g建立数据库后,向数据库中导入了部分数据,第二天继续向数据库中导入数据表时发生错误: 查了很多资料发现原来是Oracle表空间限制,导致无法继续导入数据 ...

  7. seaJS 简单例子,理解seaJS

    学习心得: 记得第一次学underscore的时候,去的官网(不管什么都是官网好),呼啦一长列语法,我就一个个看,看完也不知道underscore是做什么的.就是现在underscore我也用不上,学 ...

  8. 大数据通过PHP快速插入MYSQL的方法

    如果您的mysql是通过brew安装的,那么请 vi /usr/local/Cellar/mysql/5.6.23/my.cnf 将 max_allowed_packet = 64M 写入保存并重启m ...

  9. Linux的phpstudy mysql登录

    使用绝对路径登录 /phpStudy/mysql/bin/mysql -uroot -p; 设置远程登录密码 GRANT ALL PRIVILEGES ON *.* TO 'itoffice'@'%' ...

  10. java 线程安全不线程不安全

    经常看到一些类,有的说线程安全,有的说线程不安全,顿时懵逼. 线程安全不安全,主要是在多线程执行的情况下,如果由于线程之间抢占资源而造成程序的bug即为线程不安全,下面就拿arraylist 和Vec ...