Run Loop

run loop(运行循环)是与线程相关的基础架构的一部分。它是一个事件处理循环,用于调度工作和协调传入事件的接收。run loop的目的是在有工作做时让线程忙碌,并在没有工作可做时让线程进入休眠状态。

run loop管理不是完全自动的,必须设计线程代码以便在适当的时间启动run loop并响应传入的事件。Cocoa和Core Foundation都提供了run loop对象来帮助我们配置和管理线程的run loop。应用程序不需要明确创建run loop对象,每个线程(包括应用程序的主线程)都有一个关联的run loop对象。但是,只有辅助线程需要显式运行其run loop。作为应用程序启动过程的一部分,应用程序框架自动设置并在主线程上运行run loop。

以下内容提供了有关run loop的更多信息以及如何为应用程序配置run loop。有关run loop对象的更多信息,请参看NSRunLoop Class ReferenceCFRunLoop Reference

Run Loop详解

run loop是一个线程进入循环,使用它来运行事件处理程序以便响应传入的事件。我们的代码提供了用于实现run loop的实际循环部分的控制语句——换句话说, 我们的代码提供了驱动run loop的while或者for循环。在循环中,使用run loop对象“执行”用于接收事件和调用已安装的处理程序的事件处理代码。

run loop从两种不同类型的源中接收事件。输入源传递异步事件,这些事件通常是来自另一个线程或不同应用程序的消息。定时器源传递在预定的时间或者重复的间隔发生的同步事件。这两种类型的源都使用应用程序特定的处理程序来处理到达的事件。

下图显示了run loop和各种源的概念上的结构。输入源传递异步事件给对应的处理程序,并导致runUntilDate:方法(在线程关联的NSRunloop对象上调用)退出。定时器源传递事件到其处理例程,但是不会导致run loop退出。

Structure of a run loop and its sources.png

除了处理输入的源之外,run loop还会生成有关run loop行为的通知。已注册的run loop观察者能够接收通知,并使用它们对线程执行附加处理。使用Core Foundation来在线程上安装run loop观察者。

以下部分提供了与run loop的组件和它们的运转模式相关的更多信息,还描述了处理事件期间在不同时间生成的通知。

Run Loop模式

run loop模式是要监听的输入源和定时器的集合以及要通知的run loop观察者的集合。每次运行run loop时,都要明确或隐式地指定要运行的特定“模式”。在运行循环过程中,只监听与该模式有关的源并允许其传递事件。(同样,只有与该模式相关的观察者才会收到run loop的进度的通知。)与其他模式关联的源会保留任何新事件,直到随后以适当的模式通过循环为止。

在代码中,可以通过名称来识别模式。Cocoa和Core Foundation都定义了一个默认模式和几个常用模式,以及用于在代码中指定这些模式的字符串。可以通过简单地为模式名称指定一个自定义字符串来自定义模式。虽然分配给自定义模式的名称是任意的,但这些模式的内容不是。必须确保将一个或多个输入源、 定时器或run loop观察者添加到为它们创建的任何模式中,使它们生效。

可以使用模式在通过run loop的特定关口期间过滤掉不需要的源中的事件。大多数情况下,需要在系统定义的“默认”模式下运行run loop。但是,modal panel可能会以“模态”模式运行run loop。在此模式下,只有与modal panel相关的源才会将事件传递给线程。对于辅助线程,可以使用自定义模式来防止低优先级的源在时间至关重要的操作期间传递事件。

注意:模式基于事件源来进行区分,而不是事件的类型。例如,不会使用模式来仅仅匹配鼠标按下事件或者仅匹配键盘事件。可以使用模式来监听不同的端口集、 暂时暂停定时器或者更改当前正在监听的源和run loop观察者。

下表列出了Cocoa和Core Foundation定义的标准模式以及何时使用该模式的说明。名称列列出了用于在代码中指定模式的实际常量。

Mode Name Description
Default NSDefaultRunLoopMode(Cocoa)
kCFRunLoopDefaultMode(Core Foundation)
默认模式是用于大多数操作的模式。大多数情况下,应该使用此模式启动run loop和配置输入源。
Connection NSConnectionReplyMode (Cocoa) Cocoa将此模式与NSConnection对象一起使用来监听应答。很少需要自己使用这种模式。
Modal NSModalPanelRunLoopMode (Cocoa) Cocoa使用这种模式来识别用于modal panel的事件。
Event tracking NSEventTrackingRunLoopMode (Cocoa) Cocoa使用这种模式来限定在鼠标拖拽循环和其他类型的用户界面跟踪循环期间传入的事件。
Common modes NSRunLoopCommonModes (Cocoa)
kCFRunLoopCommonModes (Core Foundation)
这是一个常用模式的可配置组。将输入源与此模式相关联也会将其与组中的每个模式相关联。对于Cocoa应用程序,默认情况下,此集合包含默认、 模态和事件跟踪模式。Core Foundation最初只包含默认模式。可以使用CFRunLoopAddCommonMode函数将自定义模式添加到该集合中。

输入源

输入源异步传递事件给线程。事件源取决于输入源的类型,它通常是两类中的一类。基于端口的输入源监听应用程序的Mach端口。自定义输入源监听自定义事件源。对run loop而言,输入源是基于端口还是自定义的是无所谓的。系统通常实现两种类型的输入源,可以按照原样使用它们。两种源之间的唯一区别是它们如何发出信号。基于端口的源由内核自动发出信号,自定义源必须从另一个线程手动发生信号。

创建输入源时,可以将其分配给run loop的一个或者多个模式。模式能够影响在任何特定时刻哪些输入源会被监听。大多数情况下,在默认模式下运行run loop,但是也可以指定在自定义模式下运行。如果输入源不会被当前模式监听,它产生的任何事件都会被保留,直到run loop以正确的模式运行。

以下各节描述了一些输入源。

基于端口的源

Cocoa和Core Foundation为使用与端口相关的对象创建的基于端口的输入源提供了内置支持。例如,在Cocoa中,根本不需要直接创建输入源。只需要创建一个端口对象,并使用NSPort类的方法将该端口添加到run loop。端口对象为我们处理所需输入源的创建和配置。

在Core Foundation中,必须手动创建端口及其run loop源。在这两种情况下,都使用与端口不透明类型关联的函数(CFMachPortRefCFMessagePortRef或者CFSocketRef)来创建对应的对象。

有关如何设置和配置基于自定义端口的源的示例,请参看配置基于端口的输入源

自定义输入源

要创建自定义输入源,必须使用Core Foundation中的与CFRunLoopSourceRef不透明类型相关联的函数。可以使用多个回调函数来配置自定义输入源,Core Foundation会在不同的地方调用这些函数来配置该输入源,处理所有传入的事件以及在从run loop中移除该输入源时销毁该输入源。

除了在事件到达时定义自定义源的行为之外,还必须定义事件传递机制。输入源的此部分在一个单独的线程上运行,负责为输入源提供数据,并在数据准备好处理时用信号通知它。事件传递机制由我们决定,但不必过于复杂。

有关如何创建自定义输入源的示例,请参看定义自定义输入源。有关自定义输入源的参考信息,请参看CFRunLoopSource Reference

Cocoa执行选择器源(Cocoa Perform Selector Sources)

除了基于端口的源之外,Cocoa还定义了一个自定义输入源,允许我们在任何线程上执行选择器(Selector)。与基于端口的源一样,执行选择器请求在目标线程上被序列化,从而避免了在一个线程上运行多个方法时可能引发的许多同步问题。与基于端口的源不同,执行选择器源在执行其选择器后会将自己从run loop中移除。

注意:在OS X v10.5之前,执行选择器源主要用于将消息发送到主线程,但在OS X v10.5及更高版本和iOS中,可以使用它们向任何线程发送消息。

在另一个线程上执行选择器时,目标线程必须具有激活的run loop。对于创建的线程,这意味着会等待执行选择器,直到我们明确地启动run loop。由于主线程自动启动其run loop,因此只要应用程序调用应用程序委托的applicationDidFinishLaunching:方法后,就可以在主线程上发出调用。run loop会在每次通过循环时处理所有排队的执行选择器调用,而不是在每次循环迭代期间只处理一个。

下表列出了NSObject中定义的可用于在其他线程上执行选择器的方法。由于这些方法是在NSObject中声明的,所以可以在任何有权访问Objective-C对象的线程中使用它们,包括POSIX线程。这些方法实际上不会创建一个新线程来执行选择器。

Methods Description
performSelectorOnMainThread:withObject:waitUntilDone:
performSelectorOnMainThread:withObject:waitUntilDone:modes:
在该线程的下一个run loop周期中执行应用程序主线程上的指定选择器。这些方法使我们可以选择阻塞当前线程,直到选择器被执行。
performSelector:onThread:withObject:waitUntilDone:
performSelector:onThread:withObject:waitUntilDone:modes:
在任何线程上执行指定的选择器。这些方法使我们可以选择阻塞当前线程,直到选择器被执行。
performSelector:withObject:afterDelay:
performSelector:withObject:afterDelay:inModes:
在下一个run loop周期中和可选的延迟周期之后,在当前线程上执行指定的选择器。由于会等到下一个run loop周期执行选择器,所以这些方法会从当前正在执行的代码中提供一个自动微小延迟。多个排队的选择器按照它们排队的顺序依次执行。
cancelPreviousPerformRequestsWithTarget:
cancelPreviousPerformRequestsWithTarget:selector:object:
允许我们取消使用performSelector:withObject:afterDelay:或者performSelector:withObject:afterDelay:inModes:方法发送给当前线程的消息。

有关每种方法的详细信息,请参看NSObject Class Reference

定时器源

定时器源在未来的预设时间同步传递事件给我们的线程。定时器是线程通知自己做某事的一种方式。例如,搜索输入框可以使用定时器在用户连续敲击键盘之间经过一段时间后启动自动搜索。使用此延迟时间使用户有机会在开始搜索之前尽可能多地输入所需的搜索字符串。

虽然定时器源生成基于时间的通知,但其不是实时的。与输入源一样,定时器源需要与run loop的特定模式相关联。如果定时器源不在当前正在被run loop监听的模式中,其不会触发,直到run loop 以定时器源支持的模式运行。类似地,如果定时器源在run loop处于执行处理例程的过程中触发,则定时器源会等到下一次循环时调用其处理例程。如果run loop根本没有运行,则定时器永远不会触发。

可以将定时器源配置为仅生成一次或重复生成事件。重复定时器根据预定的触发时间自动重新安排自身,而不是实际的触发时间。例如,如果定时器源预定在特定时间以及之后每隔5秒触发一次,则即使实际触发时间延迟了,预定的触发时间也会始终以原来的5秒时间间隔进行。如果触发时间延迟太多以至于错过了一个或多个预定触发时间,则定时器源在错过的时间段内仅被触发一次。在错过的时间内触发后,定时器源重新安排下一个预定触发时间。

有关配置定时器源的更多信息,请参看配置定时器源。有关参考信息,请参看NSTimer Class Reference或者CFRunLoopTimer Reference

Run Loop观察者

与当一个对应的异步或同步事件发生时就会触发的源相比,run loop观察者会在run loop本身执行过程中在特定位置触发。可以使用run loop观察者来准备好线程以处理给定的事件,或者在线程进入休眠之前准备好线程。可以将run loop观察者与run loop中的以下事件相关联:

  • 进入run loop。
  • run loop即将处理定时器时。
  • run loop即将处理输入源时。
  • run loop即将进入休眠状态时。
  • run loop被唤醒但在处理唤醒它的事件之前。
  • 退出run loop。

可以使用Core Foundation将run loop观察者添加到引用程序中。要创建run loop观察者,可以创建CFRunLoopObserverRef不透明类型的新实例。此类型会跟踪我们的自定义回调函数以及它感兴趣的活动。

与定时器类似,run loop观察者可以使用一次或者重复使用。一次性的观察者在其触发后会将其自身从run loop中移除,而重复性的观察者仍然会存在于run loop中。在创建观察者时,可以指定其是运行一次还是反复运行。

有关如何创建run loop观察者的示例,请参看配置Run Loop。有关参考信息,请参看CFRunLoopObserver

Run Loop的事件处理逻辑

每当运行线程的run loop时,它都会处理未决事件,并为任何附加的观察者生成通知。其执行这些操作的顺序非常具体,如下所示:

  1. 通知观察者已经进入runloop;
  2. 进入一个do-while循环;
  3. 通知观察者即将处理定时器源接收的事件;
  4. 通知观察者即将处理自定义输入源(非基于端口)接收的事件;
  5. 处理自定义输入源(非基于端口)接收的事件;
  6. 如果基于端口的输入源有接收到系统事件,则立即跳到第10步;否则,继续执行第7步;
  7. 通知观察者线程即将进入休眠状态;
  8. 将线程置于休眠状态,直到发生以下事件:
    • 基于端口的输入源接收到系统事件;
    • 定时器源接收到定时器事件;
    • runloop运行超时;
    • runloop被手动显式唤醒;(从另一个线程传递事件给自定义输入源时,需要手动显式唤醒runloop)
  9. 通知观察者线程刚被唤醒;
  10. 处理未决事件:
    • 如果定时器源有接收到定时器事件,则处理定时器事件;
    • 如果基于端口的输入源有接收到系统事件,则处理系统事件;
  11. 如果runloop运行超时,或者被强制停止,或者不存在输入源、定时器源和观察者了,则终止do-while循环;否则,继续do-while循环(重复执行 3 ~ 10 步骤)。
  12. 通知观察者已经退出runloop。

由于定时器和输入源的观察者通知会在那些事件实际发生之前就被发送,所以通知的时间与实际事件的事件之间可能存在间隔。如果这些事件之间的时间至关重要,则可以使用休眠和从休眠中唤醒的通知来帮助将实际事件之间的时间关联起来。

由于定时器和其他周期性事件是在运行run loop时被传递的,所以要规避会扰乱这些事件的传递的循环。会发生这种行为的典型示例就是通过进入一个循环并从应用程序中重复请求事件来实现鼠标跟踪例程。因为我们的代码会直接捕获事件而不是让应用程序正常调度这些事件,所以在鼠标跟踪例程退出并将控制权返回给应用程序之前,激活的定时器将无法触发。

使用run loop对象能够显式地唤醒run loop,其他事件也可能导致run loop被唤醒。例如,添加一个不是基于端口的输入源会唤醒run loop以便输入源能够被立即处理,而不是等待直到发生其他事件。

何时使用Run Loop?

唯一需要我们明确运行run loop的时候是为应用程序创建辅助线程时。应用程序主线程的run loop是基础架构中至关重要的一部分。因此,应用程序框架提供运行主应用程序循环并自动启动该循环的代码。iOS中的UIApplication(或者OS X中的NSApplication)的运行方法启动应用程序的主循环,作为正常启动顺序的一部分。如果使用Xcode模版项目来创建应用程序,则不应该明确地调用这些例程。

对于辅助线程,我们需要确定run loop是否是必要的。如果需要,则自行配置并启动run loop。如果使用线程执行一些长时间运行且预先确定的任务,则应该避免启动run loop。run loop适用于需要与线程进行更多交互的情况。例如,如果打算执行以下任何操作,则需要启动run loop:

  • 使用端口或者自定义输入源来与其他线程进行通信。
  • 在线程中使用定时器。
  • 使用Cocoa应用程序中的任何一种performSelector…方法。
  • 让线程执行周期任务。

如果选择使用run loop,则其配置和设置非常简单。像所有线程编程一样,应当有一个机会来在适当的情况下退出辅助线程。通过自然退出线程而不是强制终止一个线程总是更好。有关如何配置和退出run loop的信息,请参看使用Run Loop对象

使用Run Loop对象

run loop对象提供了添加输入源、定时器和run loop观察者到run loop并启动run loop的主要接口。每个线程都一个与之关联的run loop对象。在Cocoa中,这个对象是NSRunLoop类的一个实例。在底层应用程序中,它是一个指向CFRunLoopRef不透明类型的指针。

获取Run Loop对象

要获取当前线程的run loop,请使用以下选项之一:

  • 在Cocoa应用程序中,使用NSRunLoopcurrentRunLoop类方法来获取NSRunLoop对象。
  • 使用CFRunLoopGetCurrent函数。

尽管它们不是自由桥接类型,但在需要时,可以从NSRunLoop对象获取CFRunLoopRef不透明类型。NSRunLoop类定义了一个getCFRunLoop方法,该方法返回可以传递给Core Foundation例程的CFRunLoopRef类型。因为两个对象都引用同一个run loop,所以可以根据需要混合调用NSRunLoop对象和CFRunLoopRef不透明类型。

配置Run Loop

在辅助线程上运行run loop之前,必须至少添加一个输入源或定时器。如果run loop没有任何要监听的源,当我们尝试运行run loop时,它会立即退出。有关如何将源添加到run loop的示例,请参看配置Run Loop源

除了安装源之外,还可以安装run loop观察者并使用它们来监听run loop的不同执行阶段。要安装run loop观察者,需要创建一个CFRunLoopObserverRef不透明类型并使用CFRunLoopAddObserver函数将其添加到run loop。run loop观察者必须使用Core Foundation创建,即使对于Cocoa应用程序也是如此。

以下代码显示了一个将run loop观察者附加到其run loop的线程的主要例程。该示例的目的是展示如何创建run loop观察者,因此代码简单地设置了一个run loop观察者来监听所有run loop活动。基础处理例程(未显示)在run loop处理定时器请求时简单地记录了run loop活动。

- (void)threadMain
{
    // The application uses garbage collection, so no autorelease pool is needed.
    NSRunLoop* myRunLoop = [NSRunLoop currentRunLoop];

    // Create a run loop observer and attach it to the run loop.
    CFRunLoopObserverContext  context = {0, self, NULL, NULL, NULL};
    CFRunLoopObserverRef    observer = CFRunLoopObserverCreate(kCFAllocatorDefault,
    kCFRunLoopAllActivities, YES, 0, &myRunLoopObserver, &context);

    if (observer)
    {
        CFRunLoopRef    cfLoop = [myRunLoop getCFRunLoop];
        CFRunLoopAddObserver(cfLoop, observer, kCFRunLoopDefaultMode);
    }

    // Create and schedule the timer.
    [NSTimer scheduledTimerWithTimeInterval:0.1 target:self selector:@selector(doFireTimer:) userInfo:nil repeats:YES];

    NSInteger    loopCount = 10;
    do
    {
        // Run the run loop 10 times to let the timer fire.
        [myRunLoop runUntilDate:[NSDate dateWithTimeIntervalSinceNow:1]];
        loopCount--;
        
    }while (loopCount);
}

为长期存活的线程配置run loop时,最好添加至少一个输入源来接收消息。尽管仅仅附加一个定时器就能进入run loop,但是一旦定时器触发,定时器通常会失效,然后会导致run loop退出。附加重复定时器可以使run loop长时间运行,但是这意味着要定期触发定时器来唤醒线程,这实际上是另一种轮询方式。相反,输入源会等待事件发生,并让线程休眠直到发生事件。

启动Run Loop

启动run loop仅仅对于应用程序中的辅助线程才是必要的。run loop必须至少附加一个输入源或者定时器来监听。如果一个都没有,则run loop会立即退出。

启动run loop的方式包括以下几种:

  • 无条件启动。
  • 设定一个时限。
  • 在特定模式下启动。

无条件地进入run loop是最简单地选择,但也是最不可取的。无条件地运行run loop会将线程置于一个永久循环中,这使得我们很难控制run loop本身。可以添加和删除输入源和定时器,但是停止run loop的唯一方法是杀死它。在自定义模式下也是无法运行run loop的。

最好使用超时值运行run loop,而不是无条件地运行run loop。当使用超时值时,run loop会一直运行直到有事件到达或分配的时间到期。如果一个事件到达,则将该事件调度给处理程序进行处理,然后run loop退出。之后,我们的代码可以重新启动run loop来处理下一个事件。如果分配的时间到期,可以简单地重新启动run loop或者使用该时间来完成任何需要的清理工作。

除了超时值之外,还可以使用特定模式运行run loop。模式和超时值不是互斥的,可以同时使用它们来启动run loop。模式限制了传递事件到run loop的源的类型,有关模式的详细信息请参看Run Loop模式

以下代码显示了一个线程的主要入口例程的粗略版本。这个示例的关键部分显示了run loop的基本结构。本质上,我们将输入源和定时器添加到run loop,然后重复调用其中一个例程来启动run loop。每次run loop例程返回时,都会检查是否有任何可能导致退出线程的情况。该示例使用Core Foundation的run loop例程,以便它可以检查返回结果并确定run loop退出的原因。如果使用Cocoa并且不需要检查返回值,则也可以使用NSRunLoop类的方法以类似的方式来运行run loop。

- (void)skeletonThreadMain
{
    // Set up an autorelease pool here if not using garbage collection.
    BOOL done = NO;

    // Add your sources or timers to the run loop and do any other setup.

    do
    {
        // Start the run loop but return after each source is handled.
        SInt32    result = CFRunLoopRunInMode(kCFRunLoopDefaultMode, 10, YES);

        // If a source explicitly stopped the run loop, or if there are no
        // sources or timers, go ahead and exit.
        if ((result == kCFRunLoopRunStopped) || (result == kCFRunLoopRunFinished))
        
            done = YES;

        // Check for any other exit conditions here and set the
        // done variable as needed.
    }
    while (!done);

    // Clean up code here. Be sure to release any allocated autorelease pools.
}

还可以递归运行一个run loop。换句话说,可以调用CFRunLoopRunCFRunLoopRunInMode或者NSRunLoop的方法来在输入源或定时器的处理例程中启动run loop。这样做时,可以使用任何需要的run loop模式来运行嵌套run loop,包括外部run loop使用的模式。

退出Run Loop

有两种方法能使run loop在其处理事件之前退出:

  • 配置run loop以超时值运行。
  • 告知run loop退出。

如果我们能够管理超时值,那么使用超时值肯定是首选。指定超时值可以让run loop完成所有正常处理,包括在退出之前将通知发送给run loop观察者。

使用CFRunLoopStop函数显式地停止运行run loop会产生类似于超时的结果。run loop会发送出任何其余的run loop通知,然后退出。不同的是,可以在对无条件启动的run loop使用此技术。

虽然移除run loop的输入源和定时器也可能导致run loop退出,但这并不是停止run loop的可靠方法。一些系统例程会将输入源添加到run loop以处理所需的事件。由于我们的代码可能无法知道这些输入源,所有就无法移除它们,这样run loop是不会退出的。

线程安全和Run Loop对象

线程安全取决于我们使用哪个API来操作run loop。Core Foundation中的函数通常是线程安全的,可以在任何线程中调用。但是,如果正在执行更改run loop配置的操作,则尽可能在持有该run loop的线程中执行此操作。

Cocoa中NSRunLoop类并不像其在Core Foundation中的副本那样是线程安全的。如果使用NSRunLoop类来修改run loop,则应该仅仅只在持有该run loop的线程中这样做。在不同的线程中将输入源或定时器添加到run loop可能会导致代码崩溃或以意外的方式运行。

配置Run Loop源

以下部分显示了如何在Cocoa和Core Foundation中设置不同类型的输入源的示例。

定义自定义输入源

创建自定义输入源涉及到以下内容:

  • 想让输入源处理的信息。
  • 能让感兴趣的客户端知道如何与输入源联系的调度例程。
  • 能用于执行任何客户端发送的请求的处理例程。
  • 能让输入源无效的取消例程。

要创建一个自定义输入源来处理自定义信息,应该灵活设计实际的配置。调度、处理和取消例程是用于自定义输入源的关键例程。然而,输入源行为的其余部分的大部分都发生在这些处理例程之外。例如,为传递数据到输入源和将输入源的存在传达给其他线程定义机制是由我们自己决定的。

下图显示了自定义输入源的示例配置。在本示例中,应用程序的主线程保持对输入源、该输入源的自定义命令缓冲区以及安装该输入源的run loop的引用。当主线程有一个任务想要切换到工作线程时,它将命令和工作线程启动该任务所需的任何信息一起发送到命令缓冲区。(因为主线程和工作线程的输入源都可以访问命令缓冲区,所以访问必须同步)。一旦命令发送,主线程就会发送信号给输入源并唤醒工作线程的run loop。在接收到唤醒命令后,run loop会调用输入源的处理程序,它会处理在命令缓冲区中找到的命令。

Operating a custom input source.png

以下各节将解释上图中自定义输入源的实现,并展示需要实现的关键代码。

定义输入源

定义自定义输入源需要使用Core Foundation例程来配置run loop源并将其附加到run loop。虽然基础处理程序是基于C语言的函数,但是这并不妨碍我们为这些函数编写包装器并使用Objective-C或者C++来实现代码的主体。

上图中介绍的输入源使用Objective-C对象来管理命令缓冲区并与run loop进行协调。以下代码显示了这个对象的定义。RunLoopSource对象管理命令缓冲区,并使用该缓冲区接收来自其他线程的消息。以下代码还显示了RunLoopContext对象的定义,该对象实际上是一个容器对象,用于将RunLoopContext对象和run loop引用传递给应用程序的主线程。

@interface RunLoopSource : NSObject
{
    CFRunLoopSourceRef runLoopSource;
    NSMutableArray* commands;
}

- (id)init;
- (void)addToCurrentRunLoop;
- (void)invalidate;

// Handler method
- (void)sourceFired;

// Client interface for registering commands to process
- (void)addCommand:(NSInteger)command withData:(id)data;
- (void)fireAllCommandsOnRunLoop:(CFRunLoopRef)runloop;

@end

// These are the CFRunLoopSourceRef callback functions.
void RunLoopSourceScheduleRoutine (void *info, CFRunLoopRef rl, CFStringRef mode);
void RunLoopSourcePerformRoutine (void *info);
void RunLoopSourceCancelRoutine (void *info, CFRunLoopRef rl, CFStringRef mode);

// RunLoopContext is a container object used during registration of the input source.
@interface RunLoopContext : NSObject
{
    CFRunLoopRef        runLoop;
    RunLoopSource*        source;
}
@property (readonly) CFRunLoopRef runLoop;
@property (readonly) RunLoopSource* source;

- (id)initWithSource:(RunLoopSource*)src andLoop:(CFRunLoopRef)loop;
@end

虽然Objective-C代码管理输入源的自定义数据和将输入源附加到run loop所需要的基于C语言的回调函数。当将输入源实际附加到run loop时,这些函数中的第一个会被调用,如下所示。由于此输入源只有一个客户端(主线程),因此它使用RunLoopSourceScheduleRoutine函数发送消息来在该线程上向应用程序委托对象注册自己。当委托对象想要与输入源通信时,它使用RunLoopContext对象中的信息来执行此操作。

void RunLoopSourceScheduleRoutine (void *info, CFRunLoopRef rl, CFStringRef mode)
{
    RunLoopSource* obj = (RunLoopSource*)info;
    AppDelegate*   del = [AppDelegate sharedAppDelegate];
    RunLoopContext* theContext = [[RunLoopContext alloc] initWithSource:obj andLoop:rl];

    [del performSelectorOnMainThread:@selector(registerSource:) withObject:theContext waitUntilDone:NO];
}

最重要的回调例程之一是用于在输入源发送信号时处理自定义数据的回调例程。以下代码显示了与RunLoopSource对象关联的RunLoopSourcePerformRoutine回调。该函数只是将执行工作的请求发送给sourceFired方法,该方法随后会处理命令缓冲区中存在的任何命令。

void RunLoopSourcePerformRoutine (void *info)
{
    RunLoopSource*  obj = (RunLoopSource*)info;
    [obj sourceFired];
}

如果使用CFRunLoopSourceInvalidate函数将输入源从其run loop中移除,则系统将调用输入源的取消例程。可以使用此例程来通知客户端其输入源不再有效,并且应该删除对它的任何引用。以下代码显示了使用RunLoopSource对象注册的取消回调例程。该函数将另一个RunLoopContext对象发送给应用程序委托对象,但是这次会要求委托对象删除对run loop源的引用。

void RunLoopSourceCancelRoutine (void *info, CFRunLoopRef rl, CFStringRef mode)
{
    RunLoopSource* obj = (RunLoopSource*)info;
    AppDelegate* del = [AppDelegate sharedAppDelegate];
    RunLoopContext* theContext = [[RunLoopContext alloc] initWithSource:obj andLoop:rl];

    [del performSelectorOnMainThread:@selector(removeSource:) withObject:theContext waitUntilDone:YES];
}

提示:应用程序委托对象的registerSource:和removeSource:方法的代码在协调输入源的客户端中展示。

在Run Loop中安装输入源

以下代码展示了RunLoopSource类的init和addToCurrentRunLoop方法。init方法创建必须被实际附加到run loop的CFRunLoopSourceRef不透明类型。它将RunLoopSource对象本身作为上下文信息传递,以便回调例程具有指向该对象的指针。不会安装输入源到run loop直到工作线程调用addToCurrentRunLoop方法,此时将调用RunLoopSourceScheduleRoutine回调函数。一旦输入源被添加到run loop中,线程就可以运行它的run loop来等待输入源传递的事件。

- (id)init
{
    CFRunLoopSourceContext context = {0, self, NULL, NULL, NULL, NULL, NULL, &RunLoopSourceScheduleRoutine, RunLoopSourceCancelRoutine, RunLoopSourcePerformRoutine};

    runLoopSource = CFRunLoopSourceCreate(NULL, 0, &context);
    
    commands = [[NSMutableArray alloc] init];

    return self;
}

- (void)addToCurrentRunLoop
{
    CFRunLoopRef runLoop = CFRunLoopGetCurrent();
    CFRunLoopAddSource(runLoop, runLoopSource, kCFRunLoopDefaultMode);
}

协调输入源的客户端

为了让输入源生效,需要操作它并从另一个线程发送信号给它。输入源的全部要点在于将关联的线程置于休眠状态直到有事件需要处理时。这一事实使得让应用程序中其他线程能够知道输入源并有一种方式来与其通信成为了必要。

将输入源告知给客户端的一种方式是在首次安装输入源到run loop时发出注册请求。可以根据需要为输入源注册尽可能多的客户端,或者可以将其注册到某个中心机构,然后将输入源发送给感兴趣的客户端。以下代码显示了由应用程序委托对象定义的在调用RunLoopSource对象的调度函数时调用的注册方法。该方法接收RunLoopSource对象提供的RunLoopContext对象,并将其添加到源列表中。以下代码还显示了用于在从run loop中移除输入源时取消注册输入源的例程。

- (void)registerSource:(RunLoopContext*)sourceInfo;
{
    [sourcesToPing addObject:sourceInfo];
}

- (void)removeSource:(RunLoopContext*)sourceInfo
{
    id    objToRemove = nil;

    for (RunLoopContext* context in sourcesToPing)
    {
        if ([context isEqual:sourceInfo])
        {
            objToRemove = context;
            break;
        }
    }

    if (objToRemove)
        [sourcesToPing removeObject:objToRemove];
}

发送信号到输入源

在客户端传递数据给输入源后,客户端必须发送信号到输入源并唤醒输入源的run loop。发送信号到输入源让run loop知道输入源已经准备好被处理。并且由于线程可能在发送信号时处于休眠状态,所以应该始终明确地唤醒run loop。如果不这样做,可能会导致run loop延迟处理输入源。

以下代码显示了RunLoopSource对象的fireCommandsOnRunLoop方法。当客户端准备好输入源来处理被添加到缓冲区的命令时,客户端会调用此方法。

- (void)fireCommandsOnRunLoop:(CFRunLoopRef)runloop
{
    CFRunLoopSourceSignal(runLoopSource);
    CFRunLoopWakeUp(runloop);
}

注意:绝对不要尝试通过使用自定义输入源来处理SIGHUP或者其他类型的过程级信号。Core Foundation中用于唤醒线程的函数不是信号安全的,不应该在应用程序的信号处理例程中使用它们。

配置定时器源

要创建一个定时器源,需要创建一个定时器对象并将其调度到run loop中。在Cocoa中,使用NSTimer类来创建新的定时器对象,而在Core Foundation中使用CFRunLoopTimerRef不透明类型。NSTimer类只是Core Foundation的扩展,其提供了一些便利功能。例如,使用同一个方法来创建和调度定时器。

在Cocoa中,可以使用以下类方法中的一种来创建和调度定时器:

  • scheduledTimerWithTimeInterval:target:selector:userInfo:repeats:
  • scheduledTimerWithTimeInterval:invocation:repeats:

这些方法会创建定时器,并将其添加到当前线程的默认模式(NSDefaultRunLoopMode)下的run loop中。如果想要手动调度定时器,可以创建NSTimer对象,然后使用NSRunLoopaddTimer:forMode:方法将其添加到run loop中。这两种技术基本上会做相同的事情,但是提供了对于定时器配置的不同级别的控制。例如,创建一个定时器并手动将其添加到run loop中,则可以使用默认模式之外的模式来执行此操作。以下代码显示了如何使用这两种技术创建定时器。第一个定时器的初始延迟时间为1秒,但是之后每隔0.1秒定时触发一次。第二个定时器在延迟0.2秒后开发触发,然后每隔0.2秒定时触发一次。

NSRunLoop* myRunLoop = [NSRunLoop currentRunLoop];

// Create and schedule the first timer.
NSDate* futureDate = [NSDate dateWithTimeIntervalSinceNow:1.0];

NSTimer* myTimer = [[NSTimer alloc] initWithFireDate:futureDate interval:0.1 target:self selector:@selector(myDoFireTimer1:) userInfo:nil repeats:YES];

[myRunLoop addTimer:myTimer forMode:NSDefaultRunLoopMode];

// Create and schedule the second timer.
[NSTimer scheduledTimerWithTimeInterval:0.2 target:self selector:@selector(myDoFireTimer2:) userInfo:nil repeats:YES];

以下代码显示了使用Core Foundation函数配置定时器所需要的代码。虽然此示例没有在上下文结构中传递任何用户定义的信息,但我们可以使用此结构来传递定时器所需的任何自定义数据。有关此结构的内容的更多信息,请参看CFRunLoopTimer Reference

CFRunLoopRef runLoop = CFRunLoopGetCurrent();
CFRunLoopTimerContext context = {0, NULL, NULL, NULL, NULL};
CFRunLoopTimerRef timer = CFRunLoopTimerCreate(kCFAllocatorDefault, 0.1, 0.3, 0, 0,
&myCFTimerCallback, &context);

CFRunLoopAddTimer(runLoop, timer, kCFRunLoopCommonModes);

配置基于端口的输入源

Cocoa和Core Foundation都提供了用于线程或者进程之间通信的基于端口的对象。以下部分介绍如何使用几种不同类型的端口设置端口通信。

配置NSMachPort对象

要使用NSMachPort对象建立本地连接,需要创建端口对象并将其添加到主线程的run loop中。在启动辅助线程时,将该端口对象传递给辅助线程的入口函数。辅助线程可以使用该端口对象将消息发送回主线程。

实现主线程代码

以下代码显示了启动辅助工作线程的主线程代码。由于Cocoa框架为配置端口和run loop执行了许多中间步骤,所以launchThread方法的代码量明显少于其在Core Foundation的等价函数的代码量。但是,两者的行为几乎完全相同。不同的是,该方式不是将本地端口的名称发送给工作线程,而是直接发送NSPort对象。

- (void)launchThread
{
    NSPort* myPort = [NSMachPort port];
    if (myPort)
    {
        // This class handles incoming port messages.
        [myPort setDelegate:self];

        // Install the port as an input source on the current run loop.
        [[NSRunLoop currentRunLoop] addPort:myPort forMode:NSDefaultRunLoopMode];

        // Detach the thread. Let the worker release the port.
        [NSThread detachNewThreadSelector:@selector(LaunchThreadWithPort:) toTarget [MyWorkerClass class] withObject:myPort];
    }
}

为了在线程之间建立一个双向通信通道,可能需要工作线程在check-in消息中发送自己的本地端口到主线程。接收check-in消息使得主线程知道在启动辅助线程时一切顺利,并且还提供了一种方式将更多消息发送到主线程。

以下代码显示了主线程的handlePortMessage:方法。当数据到达线程自己的本地端口时,会调用此方法。当check-in消息到达时,该方法直接从端口消息中检索辅助线程的端口,并将其保存以供以后使用。

#define kCheckinMessage 100

// Handle responses from the worker thread.
- (void)handlePortMessage:(NSPortMessage *)portMessage
{
    unsigned int message = [portMessage msgid];
    
    NSPort* distantPort = nil;

    if (message == kCheckinMessage)
    {
        // Get the worker thread’s communications port.
        distantPort = [portMessage sendPort];

        // Retain and save the worker port for later use.
        [self storeDistantPort:distantPort];
        
    }else
    {
        // Handle other messages.
    }
}
实现辅助线程代码

对于辅助工作线程,必须配置该线程并使用指定的端口将消息传回主线程。

以下代码显示了设置工作线程的代码。在为线程创建一个自动释放池之后,此方法创建一个工作者对象来驱动线程执行。工作者对象的sendCheckinMessage:方法为工作线程创建一个本地端口,并将一个check-in消息发送回主线程。

+(void)LaunchThreadWithPort:(id)inData
{
    NSAutoreleasePool*  pool = [[NSAutoreleasePool alloc] init];

    // Set up the connection between this thread and the main thread.
    NSPort* distantPort = (NSPort*)inData;

    MyWorkerClass*  workerObj = [[self alloc] init];
    [workerObj sendCheckinMessage:distantPort];
    [distantPort release];

    // Let the run loop process things.
    do
    {
        [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode
        beforeDate:[NSDate distantFuture]];
    }
    while (![workerObj shouldExit]);

    [workerObj release];
    [pool release];
}

在使用NSMachPort时,本地和远程线程可以使用相同的端口对象进行线程之间的单向通信。换句话说,由一个线程创建的本地端口对象成为另一个线程的远程端口对象。

以下代码显示了辅助线程的check-in例程。此方法为将来的通信设置了自己的本地端口,然后将check-in消息发送回主线程。此方法使用LaunchThreadWithPort:中收到的端口对象作为消息的目标。

// Worker thread check-in method
- (void)sendCheckinMessage:(NSPort*)outPort
{
    // Retain and save the remote port for future use.
    [self setRemotePort:outPort];

    // Create and configure the worker thread port.
    NSPort* myPort = [NSMachPort port];
    [myPort setDelegate:self];
    [[NSRunLoop currentRunLoop] addPort:myPort forMode:NSDefaultRunLoopMode];

    // Create the check-in message.
    NSPortMessage* messageObj = [[NSPortMessage alloc] initWithSendPort:outPort receivePort:myPort components:nil];

    if (messageObj)
    {
        // Finish configuring the message and send it immediately.
        [messageObj setMsgId:setMsgid:kCheckinMessage];
        [messageObj sendBeforeDate:[NSDate date]];
    }
}

配置NSMessagePort对象

要使用NSMessagePort对象建立本地连接,不能只是简单地在线程之间传递端口对象,必须按名称获取远程消息端口。在Cocoa中实现这一点需要使用特定名称来注册本地端口,然后将该名称传递给远程线程,以便它可以获取对应的端口对象进行通信。以下代码显示了在使用消息端口的情况时,端口的创建和注册过程。

NSPort* localPort = [[NSMessagePort alloc] init];

// Configure the object and add it to the current run loop.
[localPort setDelegate:self];

[[NSRunLoop currentRunLoop] addPort:localPort forMode:NSDefaultRunLoopMode];

// Register the port using a specific name. The name must be unique.
NSString* localPortName = [NSString stringWithFormat:@"MyPortName"];

[[NSMessagePortNameServer sharedInstance] registerPort:localPort name:localPortName];

在Core Foundation中配置基于端口的输入源

本节介绍如何使用Core Foundation在应用程序的主线程和工作线程之间建立双向通信通道。

以下代码显示了应用程序主线程调用的启动工作线程的代码。首先设置一个CFMessagePortRef类型来监听来自工作线程的消息。工作线程需要端口的名称来建立连接,所以字符串值被传递给工作线程的入口函数。端口名称在当前用户上下文中通常应该是唯一的,否则可能会遇到冲突。

#define kThreadStackSize        (8 *4096)

OSStatus MySpawnThread()
{
    // Create a local port for receiving responses.
    CFStringRef myPortName;
    CFMessagePortRef myPort;
    CFRunLoopSourceRef rlSource;
    CFMessagePortContext context = {0, NULL, NULL, NULL, NULL};
    Boolean shouldFreeInfo;

    // Create a string with the port name.
    myPortName = CFStringCreateWithFormat(NULL, NULL, CFSTR("com.myapp.MainThread"));

    // Create the port.
    myPort = CFMessagePortCreateLocal(NULL, myPortName, &MainThreadResponseHandler, &context, &shouldFreeInfo);

    if (myPort != NULL)
    {
        // The port was successfully created.
        // Now create a run loop source for it.
        rlSource = CFMessagePortCreateRunLoopSource(NULL, myPort, 0);

        if (rlSource)
        {
            // Add the source to the current run loop.
            CFRunLoopAddSource(CFRunLoopGetCurrent(), rlSource, kCFRunLoopDefaultMode);

            // Once installed, these can be freed.
            CFRelease(myPort);
            CFRelease(rlSource);
        }
    }

    // Create the thread and continue processing.
    MPTaskID        taskID;
    return (MPCreateTask(&ServerThreadEntryPoint, (void*)myPortName, kThreadStackSize, NULL, NULL, NULL, 0, &taskID));
}

在安装端口并启动线程之后,在等待线程的check-in消息时,主线程会继续定期执行。当check-in消息到达时,其被调度到主线程的MainThreadResponseHandler函数,如下所示。此函数提供工作线程的端口名称并为将来的通信创建管道。

#define kCheckinMessage 100

// Main thread port message handler
CFDataRef MainThreadResponseHandler(CFMessagePortRef local, SInt32 msgid,
CFDataRef data, void* info)
{
    if (msgid == kCheckinMessage)
    {
        CFMessagePortRef messagePort;
        CFStringRef threadPortName;
        CFIndex bufferLength = CFDataGetLength(data);
        UInt8* buffer = CFAllocatorAllocate(NULL, bufferLength, 0);

        CFDataGetBytes(data, CFRangeMake(0, bufferLength), buffer);
        threadPortName = CFStringCreateWithBytes (NULL, buffer, bufferLength, kCFStringEncodingASCII, FALSE);

        // You must obtain a remote message port by name.
        messagePort = CFMessagePortCreateRemote(NULL, (CFStringRef)threadPortName);

        if (messagePort)
        {
            // Retain and save the thread’s comm port for future reference.
            AddPortToListOfActiveThreads(messagePort);

            // Since the port is retained by the previous function, release
            // it here.
            CFRelease(messagePort);
        }

        // Clean up.
        CFRelease(threadPortName);
        CFAllocatorDeallocate(NULL, buffer);
    }
    else
    {
        // Process other messages.
    }

    return NULL;
}

在配置主线程之后,剩余的唯一事情是新创建的工作线程创建其自己的端口并check in。以下代码显示了工作线程的入口函数,该函数提前主线的端口名称并使用它创建远程连接回到主线程。然后该函数为自己创建一个本地端口,在该线程的run loop中安装该端口,并向包含本地端口名称的主线程发送一个check-in消息。

OSStatus ServerThreadEntryPoint(void* param)
{
    // Create the remote port to the main thread.
    CFMessagePortRef mainThreadPort;
    CFStringRef portName = (CFStringRef)param;

    mainThreadPort = CFMessagePortCreateRemote(NULL, portName);

    // Free the string that was passed in param.
    CFRelease(portName);

    // Create a port for the worker thread.
    CFStringRef myPortName = CFStringCreateWithFormat(NULL, NULL, CFSTR("com.MyApp.Thread-%d"), MPCurrentTaskID());

    // Store the port in this thread’s context info for later reference.
    CFMessagePortContext context = {0, mainThreadPort, NULL, NULL, NULL};
    Boolean shouldFreeInfo;
    Boolean shouldAbort = TRUE;

    CFMessagePortRef myPort = CFMessagePortCreateLocal(NULL, myPortName, &ProcessClientRequest, &context, &shouldFreeInfo);

    if (shouldFreeInfo)
    {
        // Couldn't create a local port, so kill the thread.
        MPExit(0);
    }

    CFRunLoopSourceRef rlSource = CFMessagePortCreateRunLoopSource(NULL, myPort, 0);
    if (!rlSource)
    {
        // Couldn't create a local port, so kill the thread.
        MPExit(0);
    }

    // Add the source to the current run loop.
    CFRunLoopAddSource(CFRunLoopGetCurrent(), rlSource, kCFRunLoopDefaultMode);

    // Once installed, these can be freed.
    CFRelease(myPort);
    CFRelease(rlSource);

    // Package up the port name and send the check-in message.
    CFDataRef returnData = nil;
    CFDataRef outData;
    CFIndex stringLength = CFStringGetLength(myPortName);
    UInt8* buffer = CFAllocatorAllocate(NULL, stringLength, 0);

    CFStringGetBytes(myPortName, CFRangeMake(0,stringLength), kCFStringEncodingASCII, 0, FALSE, buffer, stringLength, NULL);

    outData = CFDataCreate(NULL, buffer, stringLength);

    CFMessagePortSendRequest(mainThreadPort, kCheckinMessage, outData, 0.1, 0.0, NULL, NULL);

    // Clean up thread data structures.
    CFRelease(outData);
    CFAllocatorDeallocate(NULL, buffer);

    // Enter the run loop.
    CFRunLoopRun();
}

一旦工作线程进入其run loop,发送到线程的端口的所有未来事件都将由ProcessClientRequest函数处理。该函数的实现取决于线程所执行的工作类型,在此不显示。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容