解密ios响应链的工作原理

 更新时间:2023年03月22日 16:50:53   作者:远方662  
本文将深入探讨事件响应链的工作原理,并提供?Swift?中的代码示例来帮助读者更好地理解这一概念
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

事件响应链是 iOS 开发中的一个核心概念,它描述了系统将用户交互事件传递给最适合处理该事件的对象的过程。理解事件响应链的机制对于开发高质量的应用程序至关重要。本文将深入探讨事件响应链的工作原理,并提供 Swift 中的代码示例来帮助读者更好地理解这一概念。

事件响应链的工作原理

iOS 中,事件响应链的工作原理可以简单概括为:从最上层的 UIWindow 开始,依次向下传递事件,直到找到最适合处理事件的响应者对象为止。在这个过程中,每个响应者对象都有机会处理事件。

当用户执行一个操作时,如触摸屏幕或运动设备,系统会创建一个 UIEvent 对象,并将其发送到当前的第一响应者对象。如果第一响应者对象无法处理该事件,则系统会将该事件传递给响应者链中的下一个对象,直到找到能够处理该事件的对象。如果最终没有对象能够处理该事件,则事件被系统丢弃。

以下是事件响应链的示意图:

         UIWindow
             |
       UIViewController
             |
           UIView
             |
     subviews of UIView

在这个示意图中,UIWindow 是响应者链的起点,它是所有视图的根视图。UIViewControllerUIView 都是响应者对象,它们都可以处理事件。UIViewController 可以接收和处理来自其根视图的事件,而 UIView 可以接收和处理来自其自身的事件,以及来自其子视图的事件。

响应者对象的特点

响应者对象是一种特殊类型的对象,它们实现了 UIResponder 类。响应者对象可以处理事件,可以成为第一响应者对象,并且可以将事件传递给下一个响应者对象。以下是 UIResponder 类中的一些常用方法:

  • canBecomeFirstResponder:返回一个布尔值,指示对象是否可以成为第一响应者对象。
  • becomeFirstResponder:将对象设置为第一响应者对象。
  • resignFirstResponder:放弃第一响应者对象的地位。
  • next:返回响应者链中的下一个响应者对象。

响应者对象还可以实现许多方法来处理事件,例如:

  • touchesBegan(_:with:):当用户在视图上开始触摸时调用。
  • touchesMoved(_:with:):当用户在视图上移动触摸时调用。
  • touchesEnded(_:with:):当用户在视图上结束触摸时调用。
  • touchesCancelled(_:with:):当系统取消触摸事件时调用。

自定义事件处理

Swift 中,可以通过重写 UIResponder 子类的方法来自定义事件处理。以下是一个示例代码,展示如何重写 UIView 子类的 touchesBegan 方法来处理触摸事件:

class CustomView: UIView {
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        
        // 处理触摸事件
        // ...
    }
}

在这个示例中,当用户在 CustomView 上开始触摸时,系统会调用 CustomViewtouchesBegan 方法。在此方法中,开发者可以编写自己的触摸事件处理代码。

事件传递和事件响应

事件传递和事件响应是事件响应链的两个重要环节。在事件传递阶段,系统会将事件从上往下传递,直到找到最适合处理事件的对象。在事件响应阶段,系统会将事件从下往上响应,直到事件被处理或者传递到响应者链的顶部。

在事件传递阶段,UIViewUIViewController 都有一个 hitTest( *:with:) 方法,该方法返回一个 UIView 对象。当系统接收到事件时,它会调用 hitTest(* :with:) 方法来确定最适合处理该事件的视图对象。hitTest( *:with:) 方法首先检查自己是否能够处理该事件,如果不能,它会将事件传递给其子视图,并递归调用子视图的 hitTest(* :with:) 方法,直到找到能够处理该事件的视图对象。

在事件响应阶段,系统会将事件传递到第一响应者对象,并沿着响应者链向上传递,直到事件被处理或者传递到响应者链的顶部。在这个过程中,每个响应者对象都有机会处理事件。如果某个响应者对象能够处理事件,则它将调用相应的方法来处理事件,例如 touchesBegan(_:with:) 方法。如果该对象不能处理事件,则它将调用 next 方法,将事件传递给响应者链中的下一个对象。

事件拦截

hitTest(_:with:) 方法中,我们可以检查触摸点是否在指定区域内,如果在,则返回当前视图作为拦截目标,否则返回 nil,让系统将事件传递给下一个响应者。示例代码如下:

class CustomView: UIView {
    
    override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
        if self.bounds.contains(point) {
            // 触摸点在视图内,拦截事件
            return self
        } else {
            // 触摸点不在视图内,将事件传递给下一个响应者
            return super.hitTest(point, with: event)
        }
    }
    
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        print("CustomView touchesBegan")
    }
 
}

在上述代码中,我们重写了 hitTest( *:with:) 方法,并在该方法中检查触摸点是否在视图内。如果在,则返回当前视图作为拦截目标,否则返回 nil,让系统将事件传递给下一个响应者。在 touchesBegan(* :with:) 方法中,我们打印了一条日志,以便在触摸事件发生时能够看到该方法是否被调用。

事件传递到父视图

要将事件传递到父视图,可以调用 next?.touchesBegan(touches, with: event) 方法,让父视图处理触摸事件。示例代码如下:

class CustomView: UIView {
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        
        // 处理触摸事件
        // 如果无法处理该事件,传递给父视图进行处理
        next?.touchesBegan(touches, with: event)
    }
}

class ParentView: UIView {
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        
        // 处理触摸事件
    }
}

在上面的示例中,CustomViewParentView 都是 UIView 的子类。当用户在 CustomView 上触摸时,CustomViewtouchesBegan 方法会被调用。在这个方法中,如果 CustomView 无法处理该事件,它会将该事件传递给其父视图(ParentView)进行处理。

通过 next?.touchesBegan(touches, with: event) 方法将事件传递给父视图,如果父视图能够处理该事件,它的 touchesBegan 方法将被调用。在这个方法中,可以处理触摸事件。如果父视图仍然无法处理该事件,该事件将被传递给更高级别的响应对象进行处理。

需要注意的是,当事件被传递到下一个响应对象时,会调用该对象的 touchesBegan 方法。因此,在这个方法中,可以对事件进行处理,也可以将其传递给更高级别的响应对象进行处理。

自定义事件响应链

iOS 中,每个视图都是一个 UIResponder 对象。UIResponder 是一个抽象类,它定义了响应者对象可以处理的事件类型,包括触摸事件、加速计事件、远程控制事件等。每个 UIResponder 对象都有一个 next 响应者,即下一个响应者对象。当一个事件发生时,系统会将该事件从前往后依次传递给响应者链中的对象,直到某个对象处理了该事件为止。如果没有任何对象处理该事件,则该事件将被丢弃。

我们可以通过自定义 UIResponder 子类来实现更灵活的事件处理逻辑。下面是一个简单的自定义响应者链的示例代码:

class CustomResponder: UIResponder {
    
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        print("CustomResponder touchesBegan")
        next?.touchesBegan(touches, with: event)
    }
    
}

class ViewController: UIViewController {
    
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        super.touchesBegan(touches, with: event)
        print("ViewController touchesBegan")
    }
    
}

在上面的代码中,我们定义了一个名为 CustomResponder 的自定义响应者子类。在该类中,我们重写了 touchesBegan(_:with:) 方法,并在该方法中打印了一条日志。在该方法中,我们还调用了 next?.touchesBegan(touches, with: event) 方法,将触摸事件传递给下一个响应者对象。

在 ViewController 中,我们也重写了 touchesBegan( *:with:) 方法,并在该方法中打印了一条日志。当触摸事件发生时,首先会调用 CustomRespondertouchesBegan(* :with:) 方法,并打印出 "CustomResponder touchesBegan"。然后,由于我们调用了 next?.touchesBegan(touches, with: event) 方法,系统会将触摸事件传递给 CustomResponder 的下一个响应者对象,即 ViewController。此时,系统会调用 ViewControllertouchesBegan(_:with:) 方法,并打印出 "ViewController touchesBegan"。

通过自定义响应者子类,我们可以更加灵活地处理事件,实现更复杂的事件处理逻辑。例如,我们可以在响应者链中添加多个响应者对象,根据事件类型、触摸点位置等条件来决定哪个响应者对象处理该事件。

总结

事件响应链是 iOS 开发中的一个核心概念。了解事件响应链的工作原理和实现方式,可以帮助开发者更好地理解 iOS 应用的交互模型,编写更高效、可靠的交互代码。

以下是一些事件响应链的实践建议:

  • 在处理触摸事件时,始终调用父类的 touchesBegan( *:with:)touchesMoved(* :with:)touchesEnded( *:with:)touchesCancelled(* :with:) 方法。这样可以确保事件会正确地传递到响应者链的下一个对象。
  • 如果希望在事件传递过程中拦截事件,可以重写 hitTest(_:with:) 方法,并在该方法中检查事件是否应该被拦截。
  • 如果希望将事件传递到父视图,可以调用 next?.touchesBegan(touches, with: event) 方法。
  • 尽可能避免使用 touches 属性,因为该属性在多点触控环境下会出现问题。推荐使用 allTouches 属性来获取所有触摸点。
  • 尽可能避免使用 gesture recognizer 来处理触摸事件,因为这会增加系统的负担,并可能导致意外的行为。推荐使用触摸事件处理方法来处理触摸事件。

 以上就是解密ios响应链的工作原理的详细内容,希望本篇文章可以更好地帮助大家理解 iOS 应用的交互模型。更多精彩内容请大家继续关注程序员之家,我们将进行更多关于ios内容的更新。

相关文章

  • IOS 指纹识别两种方式详解及实例

    IOS 指纹识别两种方式详解及实例

    这篇文章主要介绍了IOS 指纹识别两种方式详解及实例的相关资料,需要的朋友可以参考下
    2017-06-06
  • iOS AVCaptureSession实现视频录制功能

    iOS AVCaptureSession实现视频录制功能

    这篇文章主要为大家详细介绍了iOS AVCaptureSession实现视频录制功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • 简单实现iOS指纹解锁(TouchID)

    简单实现iOS指纹解锁(TouchID)

    这篇文章主要介绍了如何简单实现iOS指纹解锁,验证TouchID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-02-02
  • iOS中导航栏pop返回时出现黑块问题的解决方法

    iOS中导航栏pop返回时出现黑块问题的解决方法

    在iOS开发的工作当中,Push和Pop经常用于界面之间的跳转和返回。下面这篇文章主要给大家介绍了关于iOS中导航栏pop返回时出现黑块问题的解决方法,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-10-10
  • Objective-C实现冒泡排序算法的简单示例

    Objective-C实现冒泡排序算法的简单示例

    冒泡排序即是依次比较相邻的两个数,如果后面的数较小则交换到前面一个数的位置上,这里我们来看一下Objective-C实现冒泡排序算法的简单示例
    2016-06-06
  • iOS中Block的回调使用和解析详解

    iOS中Block的回调使用和解析详解

    刚刚进入iOS开发行业,发现开发中要用到大量的block回调,由此可见它的重要性。本文主要讲的是 Block 回调的使用,以及 Block 是如何实现这种神奇的回调两部分来讲的,下面来一起看看吧。
    2016-09-09
  • iOS指纹识别的简单应用

    iOS指纹识别的简单应用

    这篇文章主要为大家详细介绍了iOS指纹识别的简单应用,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • iOS?UITextView?实现类似微博的话题、提及用户效果

    iOS?UITextView?实现类似微博的话题、提及用户效果

    这篇文章主要介绍了iOS?UITextView?实现类似微博的话题、提及功能,基本思路是使用正则匹配出成对的#,再利用UITextView的富文本实现高亮效果,需要的朋友可以参考下
    2022-06-06
  • 详解IOS如何防止抓包

    详解IOS如何防止抓包

    为了防止被抓包那么就要了解抓包的原理。本文将详细介绍IOS如何防止抓包,感兴趣的同学,可以参考下。
    2021-06-06
  • iOS 本地存储NSUserDefaults封装代码

    iOS 本地存储NSUserDefaults封装代码

    下面小编就为大家分享一篇iOS 本地存储NSUserDefaults封装代码,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01

最新评论

?


http://www.vxiaotou.com