Android触摸事件传递机制初识
发布时间 - 2026-01-11 00:56:01 点击率:次前言

今天总结的一个知识点是Andorid中View事件传递机制,也是核心知识点,相信很多开发者在面对这个问题时候会觉得困惑,另外,View的另外一个难题滑动冲突,比如在ScrollView中嵌套ListView,都是上下滑动,这该如何解决呢,它解决的依据就是View事件的传递机制,所以开发者需要对View的事件传递机制有较深入的理解。
目录
- Activity、View、ViewGroup三者关系
- 触摸事件类型
- 事件传递三个阶段
- View事件传递机制
- ViewGroup事件传递机制
- 小结
Activity、View、ViewGroup三者关系
我们都知道Android中看到的页面很多是Activity组件,然后在Activity中嵌套控件,比如TextView、RelativeLayout布局等,其实这些控件的基类都是View这个抽象类,而ViewGroup也是View的子类,区别在于ViewGroup是可以当做其他子类的容器,一张关系图如下:
简单一句话,这些View控件的载体是Activity,Activity通过从DecorView开始进行绘制。
触摸事件类型
ACTION_DOWN:用户手指按下操作,往往也代表着一次触摸事件的开始。
ACTION_MOVE:用户手指在屏幕上移动,一般情况下的轻微移动都会触发一系列的移动事件。
ACTION_POINTER_DOWN:额外的手指按下操作。
ACTION_POINTER_UP:额外的手指的离开操作
ACTION_UP:用户手指离开屏幕的操作,一次抬起操作标志着一次触摸事件的结束。
在一次屏幕触摸操作中,ACTION_DOWN和ACTION_UP是必需的,ACTION_MOVE则是看情况而定,如果只是点击,那么检测到只有按下和抬起操作。
事件传递三个阶段
分发(Dispatch):事件的分发对应着dispatchTouchEvent方法,在Andorid系统中,所有的触摸事件都是通过这个方法来分发的。
java boolean dispatchTouchEvent (MotionEvent ev)
这个方法中,可以决定直接消费这个事件或者将事件继续分发给子视图处理。
拦截(Intercept):事件拦截对应着onInterceptTouchEvent方法,这个方法只有在ViewGroup及其子类中才存在,在View和Activity中是不存在的。
java boolean onInterceptTouchEvent (MotionEvent ev)
这个方法用来判断是否拦截某个事件,如果拦截了某个事件,那么在同一序列事件当中,那么这个方法不会被再次调用。
消费(Consume):事件消费对应着onTouchEvent方法。
java boolean onTouchEvent (MotionEvent event)
用来处理点击事件,返回结果表示是否消耗当前事件,如果不消耗,则在同一事件序列中,当前View无法再接收到事件
在Android系统中,拥有事件传递处理能力的有三种:
Activity:拥有dispatchTouchEvent、onTouchEvent两个方法。
ViewGroup:拥有dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent三个方法。
View:拥有dispatchTouchEvent、onTouchEvent两个方法。
View事件传递机制
这里说的View指的是除了ViewGroup之外的View控件,比如TextView、Button、CheckBox等,View控件本身就是最小的单位,不能作为其他View的容器,View拥有dispatchTouchEvent、onTouchEvent两个方法,所以这里就定义了一个继承TextView的类MyTextView,通过代码查看日志,看流程如何走。
public class MyTextView extends TextView {
private static final String TAG = "MyTextView";
public MyTextView(Context context) {
super(context);
}
public MyTextView(Context context, AttributeSet attrs) {
super(context, attrs);
}
@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;
case MotionEvent.ACTION_UP:
Log.e(TAG, "dispatchTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
Log.e(TAG, "onTouchEvent ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.e(TAG, "onTouchEvent ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.e(TAG, "onTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "onTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.onTouchEvent(event);
}
}
同时定义一个MainActivity类用来展示MyTextView,在这个Activity中,我们为MyTextView设置了点击onClick和onTouch监听,方便跟踪了解事件传递的流程。
public class MainActivity extends AppCompatActivity implements View.OnClickListener, View.OnTouchListener {
private static final String TAG = "MainActivity";
private MyTextView mTextView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTextView = (MyTextView) findViewById(R.id.my_text_view);
mTextView.setOnClickListener(this); // 设置MyTextView的点击处理
mTextView.setOnTouchListener(this); // 设置MyTextView的触摸处理
}
@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;
case MotionEvent.ACTION_UP:
Log.e(TAG, "dispatchTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
Log.e(TAG, "onTouchEvent ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.e(TAG, "onTouchEvent ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.e(TAG, "onTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "onTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.onTouchEvent(event);
}
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.my_text_view:
Log.e(TAG, "MyTextView onClick");
break;
default:
break;
}
}
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
switch(view.getId()) {
case R.id.my_text_view:
switch (motionEvent.getAction()) {
case MotionEvent.ACTION_DOWN:
Log.e(TAG, "MyTextView onTouch ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.e(TAG, "MyTextView onTouch ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.e(TAG, "MyTextView onTouch ACTION_UP");
break;
default:
break;
}
break;
default:
break;
}
return false;
}
}
查看结果:
从中可以看到,事件是从down-move-up这样顺序执行,onTouch方法优先于onClick方法调用,如果都是以super方法传递的话,最后的结果是在MyTextView的onTouchEvent方法内被消费的,如果不消费的话,则会把事件返回到它的父级去消费,如果父级也没消费,那么最终会返回到Activity中处理。
ViewGroup事件传递机制
ViewGroup作为View控件的容器存在,ViewGroup拥有dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent三个方法。同样,我们自定义一个ViewGroup,继承自RelativeLayout,实现一个MyRelativeLayout。
public class MyRelativeLayout extends RelativeLayout {
private static final String TAG = "MyRelativeLayout";
public MyRelativeLayout(Context context) {
super(context);
}
public MyRelativeLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
@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;
case MotionEvent.ACTION_UP:
Log.e(TAG, "dispatchTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "dispatchTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
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;
case MotionEvent.ACTION_UP:
Log.e(TAG, "onInterceptTouchEvent ACTION_UP");
break;
default:
break;
}
return true;
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
Log.e(TAG, "onTouchEvent ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.e(TAG, "onTouchEvent ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.e(TAG, "onTouchEvent ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.e(TAG, "onTouchEvent ACTION_CANCEL");
break;
default:
break;
}
return super.onTouchEvent(event);
}
}
查看结果:
从中可以看到触摸事件的传递顺序也是从Activity到ViewGroup,再由ViewGroup递归传递给它的子View。ViewGroup通过onInterceptTouchEvent方法对事件进行拦截,如果该方法返回true,则事件不会继续传递给子View,如果返回false或者super.onInterceptTouchEvent,则事件会继续传递给子View。在子View中对事件进行消费后,ViewGroup将不接收到任何事件。
小结
在Android系统事件中,View和ViewGroup的伪代码如下:
public boolean dispatchTouchEvent(MotionEvent ev){
boolean consume = false;
if(onInterceptTouchEvent(ev)){
consume = onTouchEvent(ev);
}
else{
consume = child.dispatchTouchEvent(ev);
}
return consume;
}
用三张图来表示Android中触摸机制的流程。
1、View内触摸事件不消费
2、View内触摸事件消费
3、ViewGroup拦截触摸事件
一些总结:
- 同一个事件序列是指从手指接触屏幕的那一刻起,到手指离开屏幕的那一刻结束。一般是以down事件开始,中间含有数量不定的move事件,最终以up事件结束。
- 正常情况下,一个事件序列只能被一个View拦截且消耗。
- 某个View一旦决定拦截,那么这个事件序列就只能由它来处理,那么同一事件序列中的其他事件都不会再交给它来处理,并且事件将重新交给它的父元素去处理,即父元素的onTouchEvent会被调用。
- 如果View不消耗除ACTION_DOWN以外的其他事件,那么这个点击事件就会消失,此时父元素的onTouchEvent并不会被调用,最终会交给Activity处理。
- ViewGroup默认不拦截任何事件。
- View中没有onInterceptTouchEvent方法。
- View的onTouchEvent默认都会被消耗,除非它是不可点击的。
- 事件传递过程是由外向内的,即事件先是传递给父元素,然后再由父元素分发给子View。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# 触摸事件
# 传递机制
# 简单讲解Android开发中触摸和点击事件的相关编程方法
# android中处理各种触摸事件的方法浅谈
# Android 的触摸事件详解及示例代码
# Android触摸事件传递图解
# Android触摸事件如何实现笔触画布详解
# Android触摸事件传递机制
# 关于Android触摸事件分发的原理详析
# 都是
# 子类
# 按下
# 是从
# 可以看到
# 那一刻
# 递归
# 它来
# 再由
# 续传
# 三个阶段
# 就会
# 是在
# 都不
# 在这个
# 也没
# 是由
# 则是
# 是指
# 这个问题
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
黑客如何通过漏洞一步步攻陷网站服务器?
青岛网站建设如何选择本地服务器?
Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知
如何自定义建站之星网站的导航菜单样式?
Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】
西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?
Win11怎么设置默认图片查看器_Windows11照片应用关联设置
如何快速搭建高效可靠的建站解决方案?
Laravel如何为API生成Swagger或OpenAPI文档
高防服务器租用指南:配置选择与快速部署攻略
laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析
Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程
Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】
Laravel用户密码怎么加密_Laravel Hash门面使用教程
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
Laravel如何使用Laravel Vite编译前端_Laravel10以上版本前端静态资源管理【教程】
如何在云服务器上快速搭建个人网站?
Laravel如何使用Eloquent进行子查询
郑州企业网站制作公司,郑州招聘网站有哪些?
Python并发异常传播_错误处理解析【教程】
UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】
魔方云NAT建站如何实现端口转发?
5种Android数据存储方式汇总
高防服务器租用如何选择配置与防御等级?
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
如何在IIS中新建站点并配置端口与物理路径?
Laravel Eloquent:优雅地将关联模型字段扁平化到主模型中
微信小程序 wx.uploadFile无法上传解决办法
如何在阿里云虚拟机上搭建网站?步骤解析与避坑指南
Linux安全能力提升路径_长期防护思维说明【指导】
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
LinuxShell函数封装方法_脚本复用设计思路【教程】
SQL查询语句优化的实用方法总结
百度输入法ai面板怎么关 百度输入法ai面板隐藏技巧
Laravel定时任务怎么设置_Laravel Crontab调度器配置
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
如何用狗爹虚拟主机快速搭建网站?
edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】
浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】
如何在橙子建站中快速调整背景颜色?
如何为不同团队 ID 动态生成多个独立按钮
如何解决hover在ie6中的兼容性问题
Laravel模型关联查询教程_Laravel Eloquent一对多关联写法
北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?
如何在阿里云香港服务器快速搭建网站?
如何在万网ECS上快速搭建专属网站?
Laravel怎么在Blade中安全地输出原始HTML内容
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?

