android 事件机制,

2024-09-07 08:48
文章标签 android 事件 机制

本文主要是介绍android 事件机制,,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

view的事件机制:

不管是DOWN,MOVE,UP都会按照下面的顺序执行:

1、dispatchTouchEvent,(其中这个方法包括了setOnTouchListener(),还有onTouch()方法

2、 setOnTouchListener的onTouchListener

3、onTouchEvent 

分析在dispatchTOucheEvent()首先判断mOnTouchListener不为null,并且view是enable的状态,然后 mOnTouchListener.onTouch(this, event)返回true,这三个条件如果都满足,直接return true ; 也就是不执行onTouchEvent(event);

(1)mOnTouchListener是否为null就的看他什么时候负值的,这个时候我们在源码中看到setOnTouchListener()

(2)ENABLED是判断当前点击的控件是否是enable的,按钮默认都是enable的,因此这个条件恒定为true。

(3) mOnTouchListener.onTouch(this, event),回调控件注册touch事件时的onTouch方法。也就是说如果我们在onTouch方法里返回                   true,就会让这三个条件全部成立,从而整个方法直接返回true。如果我们在onTouch方法里返回false,就会再去执行                                       onTouchEvent(event)方法。

   注:这里面就会出现你在代码中view.setOnTouchListener(new OnTouch(){   return  true}),导致了你onTouchEvent(event)没执行了

onTouchEvent()方法中 就会去判断你的点击事件类型 比如长按还是点击,手指的操作类型,按下,移动,抬起,(对了这个里面点用了onclick()方法,这个需要知道)

 (1) setEnable(false),CLICKABLE 都是在这个里面进行判断,false的话都会直接return false;

 (2)onclick()方法在switch()中up情况下调用(先是调用performClick() ,onclick()就在这个方法中),这个方法其实就是      我们            长写的button.setOnClickListener(new OnClickListener(....))

ViewGroup事件分发:

其实比上面的view多了一些流程,就是事件怎么传递的,怎么
 这里举个栗子,(1)公司项目bug了,老板一般先找项目经理,然后项目经理再去找你,然后你解决bug,然后告诉经理,经理就不用管了(一个正常的分发)
这里的bug就是一个事件,老板就是一个屏幕吧,项目经理就是一个viewGroup,你就是一个button,
(2)公司项目bug了,老板找项目经理,经理自己解决了bug,就不去找你了,(相当于事件  给拦截了)
(3)公司项目bug了,老板找经理,经理丢给你,你发现自己解决不了,有丢给经理,然后经理给解决了(事件回传)
带着这个栗子去理解源码:

分析:把ViewGroup看作一个View,一样执行上面view的3个循序,只是在dispatchTouchEvent()中多了一个事件拦截的步骤也就是disallowIntercept变量和onInterceptTouchEvent(ev)方法,他拦截之后就不给他下面的view或者viewGroup了,如果不拦截,就给传下去下面的view和viewGroup去处理事件 ,如果这些事件在view和viewGroup没有处理,那么会回传去,这个ViewGroup自己去处理事件,处理事件就是相当于view的事件了。。。

事件拦截,是由disallowInercept || !onInterceptTouchEven(ev)  控制的(或的关系,第二个条件取放。。不要看错了)
(1)disallowIntercept是指是否禁用掉事件拦截的功能,默认是false,也可以通过调用requestDisallowInterceptTouchEvent方法对这个值进行修改
(2)onInterceptTouchEvent方法对事件传递进行拦截,onInterceptTouchEvent方法返回true代表不允许事件继续向子View传递,返回false代表不对事件进行拦截,默认返回false。
 子view和ViewGroup把事件处理掉了 ,直接在onTouchEvent()中返回true就可以了,返回false就会回传给他的父控件,

最后说下事件拦截的一些事例(http://blog.csdn.net/lmj623565791/article/details/39102591)

(1)如何拦截

重写ViewGroup的onInterceptTouchEvent方法
@Override  public boolean onInterceptTouchEvent(MotionEvent ev)  {  int action = ev.getAction();  switch (action)  {  case MotionEvent.ACTION_DOWN:  //如果你觉得需要拦截  return true ;   case MotionEvent.ACTION_MOVE:  //如果你觉得需要拦截  return true ;   case MotionEvent.ACTION_UP:  //如果你觉得需要拦截  return true ;   }  return false;  }  
默认是不拦截的,即返回false;如果你需要拦截,只要return true就行了,这要该事件就不会往子View传递了,并且如果你在DOWN retrun true ,则DOWN,MOVE,UP子View都不会捕获事件;如果你在MOVE return true , 则子View在MOVE和UP都不会捕获事件。

原因很简单,当onInterceptTouchEvent(ev) return true的时候,会把mMotionTarget 置为null ; 

(2)如何不被拦截

如果ViewGroup的onInterceptTouchEvent(ev) 当ACTION_MOVE时return true ,即拦截了子View的MOVE以及UP事件;

此时子View希望依然能够响应MOVE和UP时该咋办呢?

Android给我们提供了一个方法:requestDisallowInterceptTouchEvent(boolean) 用于设置是否允许拦截,我们在子View的dispatchTouchEvent中直接这么写:

@Override  public boolean dispatchTouchEvent(MotionEvent event)  {  getParent().requestDisallowInterceptTouchEvent(true);    int action = event.getAction();  switch (action)  {  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;  default:  break;  }  return super.dispatchTouchEvent(event);  }  
getParent().requestDisallowInterceptTouchEvent(true);  这样即使ViewGroup在MOVE的时候return true,子View依然可以捕获到MOVE以及UP事件。

参考博客:或者想细看事件分发的话
看到一个介绍很全面的文章http://blog.csdn.net/carson_ho/article/details/54136311
http://blog.csdn.net/guolin_blog/article/details/9153747
http://blog.csdn.net/lmj623565791/article/details/39102591

http://www.open-open.com/lib/view/open1462179977450.html

这篇关于android 事件机制,的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1144619

相关文章

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

Java如何通过反射机制获取数据类对象的属性及方法

《Java如何通过反射机制获取数据类对象的属性及方法》文章介绍了如何使用Java反射机制获取类对象的所有属性及其对应的get、set方法,以及如何通过反射机制实现类对象的实例化,感兴趣的朋友跟随小编一... 目录一、通过反射机制获取类对象的所有属性以及相应的get、set方法1.遍历类对象的所有属性2.获取

MySQL中的锁和MVCC机制解读

《MySQL中的锁和MVCC机制解读》MySQL事务、锁和MVCC机制是确保数据库操作原子性、一致性和隔离性的关键,事务必须遵循ACID原则,锁的类型包括表级锁、行级锁和意向锁,MVCC通过非锁定读和... 目录mysql的锁和MVCC机制事务的概念与ACID特性锁的类型及其工作机制锁的粒度与性能影响多版本

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Spring使用@Retryable实现自动重试机制

《Spring使用@Retryable实现自动重试机制》在微服务架构中,服务之间的调用可能会因为一些暂时性的错误而失败,例如网络波动、数据库连接超时或第三方服务不可用等,在本文中,我们将介绍如何在Sp... 目录引言1. 什么是 @Retryable?2. 如何在 Spring 中使用 @Retryable

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization