深究响应者链

来源:互联网 发布:什么是自动编程系统 编辑:程序博客网 时间:2024/06/01 10:06

响应者链

概念

响应者:对用户交互动作事件进行响应的对象。

响应者链:成为处理事件的响应者的先后顺序链。(当用户点击屏幕,能都对用户交互动作事件进行响应的对象组成的链条,继承自NSResponder,响应者链能够中断.)


详解:

1、Hit-Test 机制

当用户触摸(Touch)屏幕进行交互时,系统首先要找到响应者(Responder)。系统检测到手指触摸(Touch)操作时,将Touch 以UIEvent的方式加入UIApplication事件队列中。UIApplication从事件队列中取出最新的触摸事件进行分发传递到UIWindow进行处理。UIWindow 会通过hitTest:withEvent:方法寻找触碰点所在的视图,这个过程称之为hit-test view。

hitTest 的顺序如下

1 UIApplication -> UIWindow -> Root View -> ··· -> subview

在顶级视图(Root View)上调用pointInside:withEvent:方法判断触摸点是否在当前视图内;

如果返回NO,那么hitTest:withEvent:返回nil;

如果返回YES,那么它会向当前视图的所有子视图发送hitTest:withEvent:消息,所有子视图的遍历顺序是从最顶层视图一直到到最底层视图,即从subviews数组的末尾向前遍历,直到有子视图返回非空对象或者全部子视图遍历完毕。

如果有subview的hitTest:withEvent:返回非空对象则A返回此对象,处理结束(注意这个过程,子视图也是根据pointInside:withEvent:的返回值来确定是返回空还是当前子视图对象的。并且这个过程中如果子视图的hidden=YES、userInteractionEnabled=NO或者alpha小于0.1都会并忽略);

如果所有subview遍历结束仍然没有返回非空对象,则hitTest:withEvent:返回self;

系统就是这样通过hit test找到触碰到的视图(Initial View)进行响应。

2、响应者链 (Responder Chain)

有些时候,Touch后系统通过hit test 机制找到了触碰到的Initial View,但是Initial view并没有或者无法正常处理此次Touch。这个时候,系统便会通过响应者链寻找下一个响应者,以对此次Touc 进行响应。

响应者链顺序如下:

1 Initial View -> View Controller(如果存在) -> superview -> · ··  -> rootView -> UIWindow -> UIApplication


示意图

如果一个View有一个视图控制器(View Controller),它的下一个响应者是这个视图控制器,紧接着才是它的父视图(Super View),如果一直到Root View都没有处理这个事件,事件会传递到UIWindow(iOS中有一个单例Window),此时Window如果也没有处理事件,便进入UIApplication,UIApplication是一个响应者链的终点,它的下一个响应者指向nil,以结束整个循环。

3、实际开发中常见的相关问题

在实际开发中,经常会遇到视图没有响应的情况,特别是新手会经常搞不清楚状况。

一下是视图没有响应的几个情况:

1.userInteractionEnabled=NO

2.hidden=YES;

3.alpha=0~0.01;

4.没有实现touchesBegan:withEvent:方法,直接执行touchesMove:withEvent:等方法;

5.目标视图点击区域不在父视图的Frame上 (superView背景色为clear Color的时候经常会忽略这个问题)。

在某些情景下,我们在点击子视图的时候仍然需要调用父视图的touchesBegan:withEvent:等方法,例如我们在父视图上添加了一个覆盖范围了父视图大部分面积的TableView或ScrollerView 或其他View,而我们要通过父视图的touchesBegan:withEvent:方法来收键盘。

这个时候我们可以通过UIView的类别,重写touch相关方法,代码如下:


30 -(void)setEnableNextResponder:(BOOL)enableNextResponder {

    objc_setAssociatedObject(self, &enableNextResponderKey, enableNextResponderKey, OBJC_ASSOCIATION_ASSIGN);

}

-(BOOL)enableNextResponder {

    return objc_getAssociatedObject(self, &enableNextResponderKey);

}

-(void)touchesBegan:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event{

    if (self.enableNextResponder) {

        [[self nextResponder] touchesBegan:touches withEvent:event];

        [super touchesBegan:touches withEvent:event];

    }

}

-(void)touchesEnded:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {

    if (self.enableNextResponder) {

        [[self nextResponder] touchesEnded:touches withEvent:event];

        [super touchesEnded:touches withEvent:event];

    }

}

-(void)touchesMoved:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {

    if (self.enableNextResponder) {

        [[self nextResponder] touchesMoved:touches withEvent:event];

        [super touchesMoved:touches withEvent:event];

    }

}

-(void)touchesCancelled:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {

    if (self.enableNextResponder) {

        [[self nextResponder] touchesCancelled:touches withEvent:event];

        [super touchesCancelled:touches withEvent:event];

    }

}</uitouch *></uitouch *></uitouch *></uitouch *>




上方摘自:http://www.cocoachina.com/ios/20160630/16868.html


面试题如何答:

介绍响应者链:

步骤1、简单介绍

当用户点击屏幕,能都对用户交互动作事件进行响应的对象组成的链条,继承自NSResponder,响应者链能够中断。


有些时候,touch后系统通过hit test机制找到了触碰的initial View,但是initial view并没有或者无法正常处理此次Touch。这个时候,系统就会通过响应者链寻找下一个响应者,对此次Touch进行响应。


响应者链条的顺序是:initial View—>view Controller(如果存在)—>superview  —>…..—>rootView—>UIWindow —>UIApplication

步骤2、进行升华

然后可以介绍一下hit-test机制:当用户点击屏幕,系统首先找到UIResponder,然后将touch以UIEvent的方式加入到UIApplication事件队列中。UIApplication从队列中取出最新的触摸事件分发传递到UIWindow进行处理,UIWindow会通过hitTest:withEvent:方法寻找触碰点所在的视图,这个过程称之为hit-test view。

hitTest 的顺序如下

1UIApplication -> UIWindow -> Root View -> ··· -> subview

之后顶级视图会向当前视图的所有子视图发送hitTest:withEvent:消息,然后就是响应者链的过程。如果有subview的hitTest:withEvent:返回非空对象,处理结束

步骤3、最后补充:

介绍自己如何认识响应者链或者响应者链中遇到的坑:

我入坑是因为在我的应用首页,TableVeiw最下面的两个一直都是无法点击,根据我最初了解的响应者链,应该是从上到下响应,应该是可以的,后来具体的了解了hit-test机制,目标视图点击区域不在父视图的Frame上,所以点击没有效果。

以下下是视图没有响应的几个情况:


1.userInteractionEnabled=NO

2.hidden=YES;

3.alpha=0~0.01;

4.没有实现touchesBegan:withEvent:方法,直接执行touchesMove:withEvent:等方法;

5.目标视图点击区域不在父视图的Frame上 (superView背景色为clear Color的时候经常会忽略这个问题)。







1




点击打开链接

原创粉丝点击