iOS ------ 事件响应链
响应者链
响应者链是由一系列链接在一起的响应者(UIResponser之类:UIApplication,UIViewController,UIView)注组成的。一般情况下,一条响应链开始于第一响应者,结束于application对象。如果一个响应者不能处理事件,会将事件沿着响应者链传到下一个响应者
响应者对象
在响应者链中,每个响应者对象都可以处理事件,也可以选择将事件传递给下一个响应者对象进行处理,或者直接丢弃事件。响应者链中的每个响应者对象都可以重写几个方法来处理事件,这些方法包括touchesBegan:withEvent:、touchesMoved:withEvent:、touchesEnded:withEvent:等等。
响应者对象都实现了UIResponder协议,这里的实现UIResponder协议指UIApplication、UIViewController、UIView 都继承于 UIResponder。
常见的响应者对象
- UIView:是iOS中最基本的用户界面元素,可以接收用户的触摸事件并进行相关的处理。
- UIViewController:作为MVC模式中的控制器,可以响应用户的触摸事件,同时还可以管理一个或多个视图控制器。
- UIWindow:是整个应用程序的窗口,它包含了一个或多个视图,并且是接收和处理触摸事件的最高层响应者对象。
- UIGestureRecognizer:是iOS中专门用来处理手势事件的响应者对象,包括UITapGestureRecognizer、UIPanGestureRecognizer、UILongPressGestureRecognizer等等。
- UIScrollView:是一个可以滚动的视图控件,它可以接收用户的触摸事件,并在触摸拖动时进行滚动。
- UITableView:是iOS中常用的列表视图控件,它可以显示大量的数据,并且可以处理用户的滑动、点击等事件
响应者事件
iOS 中的事件可分为:触摸事件(multitouch events)、加速计事件(accelerometer events):包括摇晃、倾斜、加速等设备运动、远程控制事件(remote control events):可以来自于耳机、锁屏界面和控制中心等,例如暂停音乐、切换歌曲等。
基本元素的了解
系统是怎么响应用户的触屏事件,这里有与用户事件相关的类,它们分别是UITouch, UIEvent和UIResponder
UIResponder(事件响应者)
UIResponder是iOS中的一个基类,定义了一些接口,用于处理触摸事件和键盘事件。所有能够接受并处理事件的对象都继承于UIResponder
- UIResponder内部提供了以下方法来处理事件
// 一根或者多根手指开始触摸view,系统会自动调用view的下面方法
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
// 一根或者多根手指在view上移动,系统会自动调用view的下面方法(随着手指的移动,会持续调用该方法)
- (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
// 一根或者多根手指离开view,系统会自动调用view的下面方法
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
// 触摸结束前,某个系统事件(例如电话呼入)会打断触摸过程,系统会自动调用view的下面方法[可选]
- (void)touchesCancelled:(nullable NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
通过重写UIresponder中定义的方法,开发者可以自己类中处理用户事件,并做出相应的事件
UITouch(触摸)
- UITouch是触摸对象,一个手指一次触摸屏幕就会生成一个UITouch对象
- 若两个手指先后触摸同一个位置,第一次触摸时生成一个UITouch对象,第二次触摸更新UITouch对象的tapCount属性值由1变成2;如果两个手指一前一后触摸的位置不同,将会生成两个UITouch对象,两者没有联系。
- 每个UITouch对象会记录触摸的一些记录,包括触摸时间、位置、阶段、所处的视图和窗口等信息。 当手指移动时,系统会更新同一个UITouch对象,使之能够保存该手指的触摸位置;当手指离开屏幕上时,系统会销毁响应的UITouch对象
- UITouch对象会在触摸事件的过程中不断更新,直到触摸事件结束。在触摸事件的过程中,系统会不断向响应链的响应者发生事件,并将触摸事件封装成UIEvent对象进行传递。当触摸事件结束时,系统就会销毁相应的UITouch对象
UITouch的属性
// 记录了触摸事件产生或变化时的时间,单位是秒 The relative time at which the acceleration event occurred(read-only)
@property(nonatomic,readonly) NSTimeInterval timestamp;
// 当前触摸事件所处的状态
@property(nonatomic,readonly) UITouchPhase phase;
// touch down within a certain point within a certain amount of timen 短时间内点按屏幕的次数,可以根据tapCount判断单击、双击或更多的点击
@property(nonatomic,readonly) NSUInteger tapCount;
@property(nonatomic,readonly) UITouchType type NS_AVAILABLE_IOS(9_0);
// 触摸产生时所处的窗口
@property(nullable,nonatomic,readonly, strong) UIWindow *window;
// 触摸产生时所处的视图
@property(nullable,nonatomic,readonly, strong) UIView *view;
// The gesture-recognizer objects currently attached to the view.
@property(nullable,nonatomic,readonly,copy) NSArray <UIGestureRecognizer *> *gestureRecognizers
UITouch方法
/*返回值表示触摸在view上的位置
这里返回的位置是针对view的坐标系的(以view的左上角为原点(0, 0))
调用时传入的view参数为nil的话,返回的是触摸点在UIWindow的位置*/
- (CGPoint)locationInView:(nullable UIView *)view;
// 该方法记录了前一个触摸点的位置
- (CGPoint)previousLocationInView:(nullable UIView *)view;// Use these methods to gain additional precision that may be available from touches.
// Do not use precise locations for hit testing. A touch may hit test inside a view, yet have a precise location that lies just outside.//获取指定视图上的精确触摸位置,该方法会考虑到多点触控时不同触点之间的偏移。
- (CGPoint)preciseLocationInView:(nullable UIView *)view API_AVAILABLE(ios(9.1));
// 获取指定视图上上一次触摸的精确位置。
- (CGPoint)precisePreviousLocationInView:(nullable UIView *)view API_AVAILABLE(ios(9.1));
UIEvent(事件)
UIEvent 是 iOS 中用于表示触摸事件的类,一个 UIEvent 对象包含了所有与触摸事件相关的信息,比如触摸的位置、时间、阶段,以及多点触控时不同触点之间的状态等等。UIEvent 对象是由系统自动创建和管理的,通常情况下不需要手动创建。每产生一个事件,就会产生一个 UIEvent 对象,UIEvent 称为事件对象。
事件类型属性
//事件类型,枚举值包括触摸、运动、遥控等。
@property(nonatomic,readonly) UIEventType type NS_AVAILABLE_IOS(3_0);// 事件子类型,对于触摸事件,其子类型包括touch down、touch move、touch up等。
@property(nonatomic,readonly) UIEventSubtype subtype NS_AVAILABLE_IOS(3_0);
产生时间的事件属性
事件发生的时间戳,单位为秒。
@property(nonatomic,readonly) NSTimeInterval timestamp;
事件的传递和响应

- 步骤一:寻找目标,在iOS的视图层次结构中找到事件的最终接受者
- 步骤二:事件响应·,基于iOS响应者链处理触摸事件
事件的传递:寻找事件的第一响应者(Hit_Testing)
当一个事件发生时,事件会从父控件传给子控件
也就是说由
- 硬件 -> 系统 ->
UIApplication->UIWindow->SuperView->SubView
以上就是事件的传递,也就是寻找第一响应者的过程。
符合第一响应者的条件包括:
- touch事件的位置在响应者区域内 pointInside:withEvent: == YES
- 响应者 self.hidden != NO
- 响应者 self.alpha > 0.01
- 响应者 self.userInteractionEnabled = YES
- 遍历 subview 时,是从上往下顺序遍历的,即 view.subviews 的 + + + lastObject 到 firstObject 的顺序,找到合适的响应者view,即停止遍历.
第一响应者对于接收到的事件的三种操作:
- 不拦截,默认操作。事件会自动沿着默认的响应者链往下传递
- 拦截,不再往下分发事件。重写
touchesBegan:withEvent:进行事件处理,不调用父类的touchesBegan:withEvent: - 拦截,继续往下分发事件。重写
touchesBegan:withEvent:进行事件处理,同时调用父类的touchesBegan:withEvent:将事件往下传递
事件的响应:一旦事件的第一响应者确定了,这个事件的响应链就确定了
下图是官网对于响应者链的实例展示

每个响应者对象(UIResponder)对象都有一个nextResponder方法,用于获取响应者链中当前对象的下一个响应者。
- 图中虚线箭头是指若该
UIView是作为UIViewController根视图存在的,则其nextResponder为UIViewController对象; - 若是直接add在
UIWindow上的,则其nextResponder为UIWindow对象。
若触摸发生在UITextField上,则事件的传递顺序是:
UITextField——>UIView——>UIView——>UIViewController——>UIWindow——>UIApplication——>UIApplicationDelegation
虽然两个传递过程都设计到父子控件的传递,但它们的传递顺序和目的不同,触摸事件的传递过程主要是为了找到最合适的空间来处理事件,而响应者链传递过程是为了让控件的响应者对象能够逐级处理事件。
事件的生命周期
手指触摸屏幕的一刻,系统会生成一个触摸事件。经过IPC进程间通信,事件最终被传递给了合适的应用。
(一)系统响应阶段
- 屏幕感应到触碰后,将事件交给IOKit处理,IOKit是监测硬件的框架。IOKit将触摸事件封装成一个IOHIDEvent对象,并通过mach port传递给SpringBoard进程。
mach port是进程端口,各个进程之间通过它进行通信;
SpringBoard.app是一个系统进程,可以理解为桌面系统,可以统一管理和分发系统接收到的触摸事件;
- SpringBoard.app进程收到触摸事件,触发主线程RunLoop的source1事件源的回调。SpringBoard.app会根据当前桌面的状态,判断应该由谁响应此次触摸事件。如果没有APP在运行,则由SpringBoard处理该事件;如果有APP在运行,则由APP处理该事件;
(二)APP响应阶段
- APP进程的mach port接收到SpringBoard进程传递来的触摸事件,主线程的RunLoop被唤醒,触发source1回调;
- source1回调触发了一个source0回调,将接收到的IOHIDEvent对象封装成UIEvent对象;
- source0回调内部将触摸事件添加到UIApplication对象的事件队列中。事件出队列后,UIApplication开始寻找一个最佳响应者的过程,这个过程又称为hit-testing,具体细节在第二个主题寻找最佳响应者中阐述;
- 找到最佳响应者后,事件就在响应链中传递和响应,这里涉及到“事件的响应和响应链中的传递”;
- 经过上述流程,触摸事件要么被某个响应对象捕获后释放,要么没有找到能响应的对象被释放;
总结:触摸事件从触屏产生后,有IOKit将触摸事件传递给SpringBoard进程,再由SpingBoard分发给当前前台APP处理,触发事件响应者链事件。
完整的触摸过程
一个完整的触摸事件流程通常包括以下几个步骤:
- 手指触摸到屏幕,系统会创建一个与手指相关联的
UITouch对象,并将其加入到系统中的事件队列中。 - 系统会将该事件发送给当前
UIWindow对象,即调用UIWindow对象的touchesBegan(_:with:)方法,并将该事件传递给子视图。 - 从根视图开始,系统会通过递归调用
hitTest(_:with:)方法,寻找响应该事件的视图。在每个视图中,系统都会调用point(inside:with:)方法,判断该视图是否包含该事件的触摸点。 - 一旦找到了响应该事件的视图,系统会将该事件发送给该视图,即调用该视图的
touchesBegan(:with:)方法。 - 在该视图的
touchesBegan(:with:)方法中,开发者可以对该事件做出相应的处理,比如更改视图的状态、更新视图的内容等。 - 如果该事件需要传递给其它视图进行处理,开发者可以手动调用
next方法,将该事件传递给下一个响应者。 - 当手指离开屏幕时,系统会将一个
touch对象的phase属性设置为.ended,并将该touch对象从事件队列中移除。 - 当前的
UIWindow对象会将该事件发送给响应者链中的下一个响应者。如果没有下一个响应者,则该事件的响应过程结束。
总结:
- 当触摸事件发生后,系统会自动生成一个
UIEvent对象,记录事件发生的事件和类型 - 然后系统会把
UIEvent事件加入到一个由UIApplocation管理的事件队列中 - 然后
UIApplication会讲事件分发给UIWindow,主窗口会在视图层次结构中找到一个合适的响应者对象来处理触摸事件。 - 不断递归调用
hitTest方法来找到第一响应者 - 如果第一响应者无法响应事件,那么会按照响应者链往上传递,也就是传递给自己的父视图
- 一直传递直到
UIApplication,如果都无法响应,事件就会被丢弃
相关文章:
iOS ------ 事件响应链
响应者链 响应者链是由一系列链接在一起的响应者(UIResponser之类:UIApplication,UIViewController,UIView)注组成的。一般情况下,一条响应链开始于第一响应者,结束于application对象。如果一个…...
Go 语言 switch 语句的特点
在 Go 语言中,switch 语句设计得更加简洁和直观,因此不需要显式使用 break 语句来终止一个分支。这种设计决策源于 Go 语言的一些设计哲学和目标,主要包括: 自动终止: Go 语言的 switch 语句会在每个 case 执行完成后自…...
【递归】什么是递归-C语言为例
递归是指一个函数在其定义中直接或间接调用自身的编程技巧。在C语言中,递归常用于解决可以被分解为更小的子问题的问题。递归函数通常由两个主要部分组成: 基准情况:这是递归停止的条件,通常是最简单的情况。 递归情况࿱…...
vue针对低版本浏览器不兼容es6特性解决方案,
browser.min.js 解决ES6兼容IE browser.min.js,polyfill.min.js vue针对安卓低版本、ios9 不兼容 es6特性解决方案 解决IE9无法使用promise的js脚本,引入后,还需跟browser.js配合使用 Babel 默认只转换新的 JavaScript 句法,po…...
嵌入式内存管理高频面试题及参考答案(4万字长文)
目录 嵌入式系统中内存管理的重要性 嵌入式系统中的内存主要分为哪几类? 静态内存分配和动态内存分配的特点 内存对齐的概念及其作用 嵌入式系统中为什么需要关注内存碎片问题 内存分区的概念及其在嵌入式系统中的应用 内存映射文件的概念及其在嵌入式系统中的作用 虚…...
TinyWebserver的复现与改进(2):项目的整体框架
上文我们成功运行了代码,本文我们将对项目的整体流程作一下讲解 如果你之前没做过相关的内容,对服务器的⾼并发模型也⼀⽆所知,不建议继续做下去,需要的前置知识有: Linux的基本命令(⭐)多进程…...
R 语言学习教程,从入门到精通,R 字符串(10)
1、R 字符串 R 语言字符串可以使用一对单引号 ’ ’ 或一对双引号 " " 来表示。 单引号字符串中可以包含双引号。 单引号字符串中不可以包含单引号。 双引号字符串中可以包含单引号。 双引号字符串中不可以包含双引号。 以下示例演示来字符串的使用: a …...
QT 简易音乐播放器
目录 放置控件 获取mp3文件 播放音乐 准备工作 加载模块 加载头文件 new一个output对象,Mediaplayer对象 把outpout对象交给mediaplayer对象 给播放器设置音乐 播放 优化 上一曲下一曲功能 双击歌曲播放 获取音乐时长和音乐播放时间 让音乐进度条跟随音乐走 调…...
代码随想录八股训练营day32
代码随想录八股训练营day32 1、synchronized和lock的区别是什么 (1)synchronized和lock的区别是什么 synchronized和Lock都是Java中用于实现线程同步的手段,synchronized是Java的关键字,基于JVM的内置锁实现,可以用于…...
11.面试题——消息队列RabbitMQ
1.RabbitMQ是什么?特点是什么? RabbitMQ是一种开源的消息队列中间件,用于在应用程序之间进行可靠的消息传递。它实现了AMQP(Advanced Message Queuing Protocol)协议,提供了强大的消息处理能力。RabbitMQ的…...
MySQL运维-日志
错误日志 二进制日志 介绍 日志格式 日志查看 日志删除 查询日志 慢查询日志...
synchronized重量级锁的实现原理是怎样的
重量级锁(Heavyweight Locking)是 Java 中 synchronized 关键字的锁机制的一部分,用于在高竞争情况下确保线程的同步。重量级锁主要通过操作系统的线程同步机制实现,通常涉及阻塞线程、上下文切换等开销较大的操作。以下是重量级锁…...
探索 GLTF 的世界:3D 内容的未来
在 3D 内容创作领域,GLTF 正在掀起波澜,成为跨不同平台提供丰富互动体验的未来标准。GL 传输格式 (GLTF) 由 Khronos Group 开发,是一种用于在工具和服务之间传输 3D 模型和场景的开放标准。它设计紧凑、高效且易于集成…...
【深度学习】【语音】TTS, CM-TTS,TTS扩散模型,论文
CM-TTS: Enhancing Real Time Text-to-Speech Synthesis Efficiencythrough Weighted Samplers and Consistency Models CM-TTS: 提高实时文本到语音合成效率 通过加权采样器和一致性模型 Xiang Li 1, Fan Bu 1, Ambuj Mehrish 2, Yingting Li 1, Jiale Han 1, Bo Cheng 1, S…...
【网络协议】网络劫持 - ARP_DNS欺骗篇
前言 网络劫持是一种网络攻击技术,攻击者通过拦截、篡改或重定向数据流量,控制用户的网络通信路径,干扰正常的网络服务。其方式可能包括DNS劫持、ARP欺骗和HTTP劫持等。通过这些手段,攻击者可以窃取敏感信息如个人身份数据和财务信…...
Linux 系统框架分析(一)
一、linux内核结构框图 对内核结构框图有个总体的把握,有助于理解为什么驱动要这样写,为什么写的应用程序所用的C库接口能够产生这么多的事情。 框图可以看出来,linux系统,包括五个系统 一、Linux内核结构介绍 Linux 内核是操作…...
Leetcode 剑指 Offer II 090.打家劫舍 II
题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer(专项突击版)系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 一个专业的小偷,计划偷窃一个环形街道上沿街的房屋&a…...
上海冷链配送新篇章 华鼎冷链科技以卓越服务餐饮品牌
在快速发展的上海餐饮连锁行业中,冷链运输作为保障食品安全与品质的关键环节,正迎来前所未有的发展机遇与挑战。华鼎冷链科技作为该领域的佼佼者,正引领着上海乃至全国冷链运输行业的新风尚。 华鼎冷链科技的成功并非一蹴而就。首先ÿ…...
学习鸿蒙-应用市场申请签名
1.需要的文件概念 .cer / .p7b / .p12 / .csr HarmonyOS应用/服务通过数字证书(.cer文件)和Profile文件(.p7b文件)来保证应用/服务的完整性。在申请数字证书和Profile文件前,首先需要通过DevEco Studio来生成密钥&am…...
LayUi插件
文档:日期和时间组件文档 - Layui layDate安装 npm install layui-laydate...
钨金属与钢在氩气环境中COMSOL全耦合电弧-等离子体-熔池交互过程研究
comsol电弧-等离子体-熔池全耦合 钨金属和钢在氩气环境中作用电弧焊接中的金属相变就像一场高温芭蕾——钨电极引燃的等离子体焰流在氩气保护下亲吻钢板,瞬间将固态金属熔化为液态舞池。今天我们用COMSOL复现这场热力秀,看看当3000K的钨遇上1500℃的钢&a…...
解锁Visual Studio中的图标编辑:.CUR文件的编辑指南
在软件开发中,图标是用户界面设计的重要组成部分。它们不仅能增强应用程序的美观度,还能提供直观的操作指引。然而,对于那些不熟悉Visual Studio环境的开发者来说,编辑图标文件可能遇到一些障碍。本文将详细介绍如何在Visual Studio中编辑.CUR文件,以及为什么默认情况下这…...
OpenClaw更新操作
文章名称 目录文章名称前言一、OpenClaw更新26.3.31版本二、飞书更新26.3.31版本我的龙虾日记前言 OpenClaw由于每个版本都有大量内容,更新的时候会出很多问题。记录一下出现过的问题 一、OpenClaw更新 推荐采用重装的方式进行更新,由于会进行新手教程.如果你不想再…...
Godot资源解压器godotdec:从游戏资源保护到开发分析的技术实践
Godot资源解压器godotdec:从游戏资源保护到开发分析的技术实践 【免费下载链接】godotdec An unpacker for Godot Engine package files (.pck) 项目地址: https://gitcode.com/gh_mirrors/go/godotdec 在游戏开发与资源管理领域,Godot引擎的.pck…...
Featurize深度学习训练全流程解析:从数据上传到模型输出
1. 数据上传:从本地到云端的高效迁移 第一次使用Featurize上传数据集时,我习惯性地点开了网页端的上传按钮,结果发现系统自动启用了分片上传机制。这个细节让我印象深刻——当我的10GB图像数据集在上传过程中网络波动时,竟然不需要…...
LM1875电路调校实战:从元件选型到稳定性优化全解析
1. LM1875功放电路基础解析 LM1875作为经典的音频功放芯片,以其结构简单、音质优良著称。但很多初学者在复刻电路时容易陷入"照搬电路图却问题频出"的困境。我们先拆解官方电路图中每个元件的实际作用,这比单纯知道"用什么"更重要。…...
单片机案例:单位数码管显示0,7和轮转显示0—9
文章目录1.单位数码管显示0效果图代码2.单位数码管显示7效果图代码3.单位数码管轮转显示0—9效果图代码1.单位数码管显示0 效果图 代码 #include <reg52.h>#define uchar unsigned char #define uint unsigned int// 定义锁存器控制引脚 sbit LE P2^7; // 74HC573的锁…...
Flink on K8s实战:从源码到部署,手把手教你自定义Job提交流程
Flink on K8s深度定制:从源码改造到生产级部署的全链路实践 1. 为什么需要自定义Flink on K8s的提交流程? 在标准的Flink on Kubernetes部署中,官方提供的客户端工具已经能够满足基础需求。但当企业面临以下场景时,原生方案就会显…...
告别手动配置:用快马AI生成openclaw自动化安装与环境管理脚本
作为一名经常需要配置各种开发环境的程序员,我深刻体会到手动安装工具的繁琐。最近在搭建一个爬虫项目时需要用到openclaw,发现传统安装方式存在几个明显的效率痛点: 版本查找耗时:需要反复在官网和文档间切换,确认最…...
TOAST UI Chart折线图实战:实时数据更新与同步工具提示完整指南
TOAST UI Chart折线图实战:实时数据更新与同步工具提示完整指南 【免费下载链接】tui.chart 🍞📊 Beautiful chart for data visualization. 项目地址: https://gitcode.com/gh_mirrors/tu/tui.chart TOAST UI Chart是一款功能强大的数…...
