iOS--runloop的初步认识

runloop的初步认识

  • 简单认识runloop
    • Event loop
    • runloop其实就是个对象
      • NSRunloop和CFRunLoopRef的依赖关系
      • runloop与线程
      • runloop mode
      • runloop source
        • CFRunLoopSource
        • CFRunLoopObserver
        • CFRunLoopTimer
    • runloop的实现
      • runloop的获取
      • 添加Mode
        • CFRunLoopAddCommonMode
      • 添加Run Loop Source(ModeItem)
    • runloop运行
      • CFRunLoopRunSpecific
    • __CFRunLoopRun
    • __CFRunLoopServiceMachPort
    • runloop的应用
      • 线程保活

简单认识runloop

runloop是什么,Runloop,即运行循环,是苹果操作系统(如iOS、macOS)中一个重要的基础架构组件,它是线程相关的,负责管理和调度线程上的各种事件和任务,确保线程在有工作时忙碌,无工作时休眠,以此达到高效利用系统资源的目的。

这里引用一下苹果文档的解释:

RunLoop是与线程息息相关的基本结构的一部分。RunLoop是一个调度任务和处理任务的事件循环。RunLoop的目的是为了在有工作的时候让线程忙起来,而在没有工作的时候让线程进入休眠状态。

这里举一个最常见的例子,我们打开xcode中的一个项目,找到main文件:

#import <UIKit/UIKit.h>
#import "AppDelegate.h"int main(int argc, char * argv[]) {NSString * appDelegateClassName;@autoreleasepool {// Setup code that might create autoreleased objects goes here.appDelegateClassName = NSStringFromClass([AppDelegate class]);}return UIApplicationMain(argc, argv, nil, appDelegateClassName);
}

这段代码的作用是保存下来该应用程序的代理类的类名,并在返回中返回了这个应用程序的主线程上的runloop,其中UIApplicationMain(argc, argv, nil, appDelegateClassName) 是启动一个iOS应用程序的中心函数调用,它在main函数中被调用,负责初始化整个应用程序环境并开始执行应用的事件循环。或者说UIApplicationMain 函数创建应用程序对象和主运行循环,并传递控制权给应用程序的委托类(AppDelegate)来处理应用程序的逻辑。
同时,UIApplicationMain函数内部帮我们开启了主线程的RunLoop。
我通过Xcode内查询声明,无法找到该函数的具体源码,但通过网上的资料可以了解到该函数内部应该是存在一个无限循环 ;

function loop() {initialize();do {var message = get_next_message();process_message(message);} while (message != quit);
}

关于这个循环,首先我们要了解一下Event loop

Event loop

Event Loop(事件循环)是许多现代编程环境和框架中的一个核心概念,特别是在需要处理并发和异步操作的场景下,如Web浏览器的JavaScript引擎、Node.js服务端环境、以及某些图形界面和操作系统中。其基本职责是持续监听和处理各种输入事件,协调和调度任务的执行,确保程序的响应性和高效运行。其基本组成包括:

  • 消息队列(Event Queue):所有外部事件(如用户输入、网络响应、计时器到期)和待处理的任务都会被放置到一个或多个消息队列中等待处理。这些队列是先进先出(FIFO)的。
  • 事件循环:这是一个持续运行的循环,其主要任务是从消息队列中取出事件或任务,然后分配给相应的处理函数执行。一旦执行完毕,循环会继续检查队列中是否有更多事件,有的话则继续处理,没有则等待。
  • 回调函数/事件处理器:当事件循环从队列中取出一个事件时,会执行与该事件关联的回调函数或事件处理器,完成具体的业务逻辑处理。

其具体的工作流程和结构在runloop中也基本体现了 ,这里就不讲了 ;、

runloop其实就是个对象

这点是很中要的,首先,在最外层我们接触到的runloop,应该是NSrunloop类对象,但往深处了解的话,其实这个类其实依赖于CFFoundation框架CFRunLoopRef类型 ;这个类我没能直接查看到源码,在网上查询资料了解到了大概的结构 ;

struct __CFRunLoop {CFRuntimeBase _base;pthread_mutex_t _lock;  __CFPort _wakeUpPort;   // used for CFRunLoopWakeUp 内核向该端口发送消息可以唤醒runloopBoolean _unused;volatile _per_run_data *_perRunData; pthread_t _pthread;             //RunLoop对应的线程uint32_t _winthread;CFMutableSetRef _commonModes;    //存储的是字符串,记录所有标记为common的modeCFMutableSetRef _commonModeItems;//存储所有commonMode的item(source、timer、observer)CFRunLoopModeRef _currentMode;   //当前运行的modeCFMutableSetRef _modes;          //存储的是CFRunLoopModeRefstruct _block_item *_blocks_head;struct _block_item *_blocks_tail;CFTypeRef _counterpart;
};

大概看一下上面有注释说明的几个成员变量,它们在之后会再重点分析 ;
一个RunLoop对象,主要包含了一个线程,若干个Mode,若干个commonMode,还有一个当前运行的Mode。

NSRunloop和CFRunLoopRef的依赖关系

[NSRunLoop currentRunLoop];//获得当前RunLoop对象
[NSRunLoop mainRunLoop];//获得主线程的RunLoop对象
//NSRunLoop是基于CFRunLoopRef的封装,提供了面向对象的API,但是这些API不是线程安全的。
CFRunLoopGetCurrent();//获得当前线程的RunLoop对象
CFRunLoopGetMain();//获得主线程的RunLoop对象
//CFRunLoopRef是在CoreFoundation框架内的,其提供了纯C语言函数的API,所有这些API都是线程安全的。

具体实现就不解释了,其实上面方法的实现就是对下面方法实现的封装 ;

然后就是分析runloop对象中的成员变量了 ;

runloop与线程

pthread_t _pthread;             //RunLoop对应的线程

首先要知道几点:

  • Runloop 和线程是绑定在一起的。每个线程(包括主线程)都有一个对应的 Runloop 对象。我们并不能自己创建 Runloop 对象,但是可以获取到系统提供的 Runloop 对象。
  • 主线程的 Runloop 会在应用启动的时候完成启动,其他线程的 Runloop 默认并不会启动,需要我们手动启动。

然后就要看看CFRunLoopGetCurrent();//获得当前线程的RunLoop对象
CFRunLoopGetMain();//获得主线程的RunLoop对象
中都会调用的函数:_CFRunLoopGet0(不过我这些方法函数实现我在xcode中无法查看具体实现,所以只能看别人公开的部分源码)

//全局的Dictionary,key是pthread_t,value是CFRunLoopRef
static CFMutableDictionaryRef __CFRunLoops = NULL;
//访问__CFRunLoops的锁
static CFSpinLock_t loopsLock = CFSpinLockInit;// should only be called by Foundation
// t==0 is a synonym for "main thread" that always works
//t==0是始终有效的“主线程”的同义词//获取pthread对应的RunLoop
CF_EXPORT CFRunLoopRef _CFRunLoopGet0(pthread_t t) {if (pthread_equal(t, kNilPthreadT)) {//pthread为空时,获取主线程t = pthread_main_thread_np();}__CFSpinLock(&loopsLock);if (!__CFRunLoops) {__CFSpinUnlock(&loopsLock);//第一次进入时,创建一个临时字典dictCFMutableDictionaryRef dict = CFDictionaryCreateMutable(kCFAllocatorSystemDefault, 0, NULL, &kCFTypeDictionaryValueCallBacks);//根据传入的主线程获取主线程对应的RunLoopCFRunLoopRef mainLoop = __CFRunLoopCreate(pthread_main_thread_np());//保存主线程,将主线程-key和RunLoop-Value保存到字典中CFDictionarySetValue(dict, pthreadPointer(pthread_main_thread_np()), mainLoop);//此处NULL和__CFRunLoops指针都指向NULL,匹配,所以将dict写到__CFRunLoopsif (!OSAtomicCompareAndSwapPtrBarrier(NULL, dict, (void * volatile *)&__CFRunLoops)) {//释放dictCFRelease(dict);}//释放mainRunLoopCFRelease(mainLoop);__CFSpinLock(&loopsLock);}//以上说明,第一次进来的时候,不管是getMainRunLoop还是get子线程的runLoop,主线程的runLoop总是会被创建//从全局字典里获取对应的RunLoopCFRunLoopRef loop = (CFRunLoopRef)CFDictionaryGetValue(__CFRunLoops, pthreadPointer(t));__CFSpinUnlock(&loopsLock);if (!loop) {//如果取不到,就创建一个新的RunLoopCFRunLoopRef newLoop = __CFRunLoopCreate(t);__CFSpinLock(&loopsLock);loop = (CFRunLoopRef)CFDictionaryGetValue(__CFRunLoops, pthreadPointer(t));//创建好之后,以线程为key,runLoop为value,一对一存储在字典中,下次获取的时候,则直接返回字典内的runLoopif (!loop) {//把newLoop存入字典__CFRunLoops,key是线程tCFDictionarySetValue(__CFRunLoops, pthreadPointer(t), newLoop);loop = newLoop;}// don't release run loops inside the loopsLock, because CFRunLoopDeallocate may end up taking it__CFSpinUnlock(&loopsLock);CFRelease(newLoop);}//如果传入线程就是当前线程if (pthread_equal(t, pthread_self())) {_CFSetTSD(__CFTSDKeyRunLoop, (void *)loop, NULL);if (0 == _CFGetTSD(__CFTSDKeyRunLoopCntr)) {//注册一个回调,当线程销毁时,销毁对应的RunLoop_CFSetTSD(__CFTSDKeyRunLoopCntr, (void *)(PTHREAD_DESTRUCTOR_ITERATIONS-1), (void (*)(void *))__CFFinalizeRunLoop);}}return loop;
}

这里还要注意几点,除了我上面说到的Runloop 和线程是绑定在一起的。

  • RunLoop保存在一个全局的Dictionary里面,线程作为key,RunLoop作为Value
  • 线程刚创建的并没有RunLoop对象。RunLoop会在第一次获取线程的RunLoop创建,在线程结束的时候销毁。
  • 主线程的RunLoop已经自动获取(创建),子线程默认没有开启RunLoop.

在这里插入图片描述

runloop mode

runloop mode是runloop中非常重要的组成部分,我的理解是它是runloop的运行模式 ;
它的结构如下:

struct __CFRunLoopMode {CFRuntimeBase _base;pthread_mutex_t _lock;  /* must have the run loop locked before locking this */CFStringRef _name;   //mode名称Boolean _stopped;    //mode是否被终止char _padding[3];//几种事件CFMutableSetRef _sources0;  //sources0CFMutableSetRef _sources1;  //sources1CFMutableArrayRef _observers; //通知CFMutableArrayRef _timers;    //定时器CFMutableDictionaryRef _portToV1SourceMap; //字典  key是mach_port_t,value是CFRunLoopSourceRef__CFPortSet _portSet; //保存所有需要监听的port,比如_wakeUpPort,_timerPort都保存在这个数组中CFIndex _observerMask;
#if USE_DISPATCH_SOURCE_FOR_TIMERSdispatch_source_t _timerSource;dispatch_queue_t _queue;Boolean _timerFired; // set to true by the source when a timer has firedBoolean _dispatchTimerArmed;
#endif
#if USE_MK_TIMER_TOOmach_port_t _timerPort;Boolean _mkTimerArmed;
#endif
#if DEPLOYMENT_TARGET_WINDOWSDWORD _msgQMask;void (*_msgPump)(void);
#endifuint64_t _timerSoftDeadline; /* TSR */uint64_t _timerHardDeadline; /* TSR */
};

一个CFRunLoopMode对象有一个name,若干source0、source1、timer、observer和若干port,可见事件都是由Mode在管理,而RunLoop管理Mode。

结构如图:
在这里插入图片描述

苹果文档中提到的 Mode 有五个,分别是:

  • NSDefaultRunLoopMode
  • NSConnectionReplyMode
  • NSModalPanelRunLoopMode
  • NSEventTrackingRunLoopMode
  • NSRunLoopCommonModes

这五个mode的使用场景和运行职责也各自不同,这里用到网上的一张图来解释:
在这里插入图片描述

我看还有一种说法是系统默认注册五个Mode:

kCFRunLoopDefaultMode:App的默认Mode,通常主线程是在这个Mode下运行
UITrackingRunLoopMode:界面跟踪Mode,用于ScrollView追踪触摸滑动,保证界面滑动时不受其他 Mode 影响(与用户交互事件的Mode)
UIInitializationRunLoopMode: 在刚启动 App 时第进入的第一个 Mode,启动完成后就不再使用,会切换到kCFRunLoopDefaultMode
GSEventReceiveRunLoopMode: 接受系统事件的内部 Mode,通常用不到
kCFRunLoopCommonModes: 这是一个占位用的Mode,作为标记kCFRunLoopDefaultMode和UITrackingRunLoopMode用,并不是一种真正的Mode (伪模式,并不是一种真正的模式)
其中kCFRunLoopDefaultMode、UITrackingRunLoopMode、kCFRunLoopCommonModes是我们开发中需要用到的模式。

个人觉得这两个其实差不多,第一个更详细具体,但第二种明显更好理解 ;

到这里也大概能理解我上面说mode是runloop的运行模式了 ;

这里要就看一下kCFRunLoopCommonModes;这个上面的解释其实也不是很清楚,其功能实现包括以下两点:

  • 一个Mode可以将自己标记为Common属性,通过将其ModeName添加到RunLoop的commonModes中。
  • 每当RunLoop的内容发生变化时,RunLoop都会将_commonModeItems里的Source/Observer/Timer同步到具有Common标记的所有Mode里。

然后也可以看看网上的源码版本,也更能理解其作用;

void CFRunLoopAddCommonMode(CFRunLoopRef rl, CFStringRef modeName) {CHECK_FOR_FORK();if (__CFRunLoopIsDeallocating(rl)) return;__CFRunLoopLock(rl);if (!CFSetContainsValue(rl->_commonModes, modeName)) {//获取所有的_commonModeItemsCFSetRef set = rl->_commonModeItems ? CFSetCreateCopy(kCFAllocatorSystemDefault, rl->_commonModeItems) : NULL;//获取所有的_commonModesCFSetAddValue(rl->_commonModes, modeName);if (NULL != set) {CFTypeRef context[2] = {rl, modeName};//将所有的_commonModeItems逐一添加到_commonModes里的每一个ModeCFSetApplyFunction(set, (__CFRunLoopAddItemsToCommonMode), (void *)context);CFRelease(set);}} else {}__CFRunLoopUnlock(rl);
}

这里可以举一个简单的例子,如果是在默认的设置下,当我们滑动某个控件时,我们在同一个线程上的定时器在滑动时停止,在定时轮播图的实现中应该都遇到过 ;这里的原因就是也就是上面说的,runloop管理mode,mode管理Source/Observer/Timer,当我们直接添加定时器的时候:

[NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
//系统会自动加入NSDefaltRunLoopMode.
//[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

在主线程中,不执行滑动操作,runloop中的mode为kCFRunLoopDefaultMode;我们此时将timer加入该mode进行管理,但当我们执行滑动时,runloop会切换mode为UITrackingRunLoopMode,而在这个mode中没有添加的timer,因此timer会停止 ;
最简单的解决方法就是,将定时器加入commonmode中管理,这样在每次切换mode时,都会将commonmode中的Source/Observer/Timer添加到新的mode中 ;

[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

如果对这个切换mode的过程不清楚,可以看看之前展示的思维导图,也就是runloop的工作流程 ;

runloop source

Run Loop Source分为Source、Observer、Timer三种,他们统称为ModeItem。

CFRunLoopSource

CFRunLoopSource分source0和source1;
结构如下:

```c
/* 
定义了一个CFRunLoopSource结构体,这是Core Foundation框架中用于处理事件源的基础结构,是Runloop能够识别和管理的事件的抽象。
*/
struct __CFRunLoopSource {CFRuntimeBase _base; // CFRuntimeBase是Core Foundation框架中所有对象共有的基类结构,包含引用计数等基本信息。uint32_t _bits; // 该字段用于标记Source的状态,特别是Signaled状态。对于source0(一次性事件源),只有当它被标记为Signaled时,Runloop才会去处理它。pthread_mutex_t _lock; // 互斥锁,保护CFRunLoopSource内部状态的线程安全,确保在多线程环境下对_source的修改是原子的。CFIndex _order;         /* immutable */ // 源的执行优先级顺序,不可变。数值越小,优先级越高,影响源在相同模式下的执行顺序。CFMutableBagRef _runLoops; // 这个字段存储了所有引用了当前source的RunLoop对象。一个source可以被多个RunLoop共享,这个bag记录了这些关联。union { // 联合体,用于存储不同版本的上下文结构,以支持向后兼容及功能扩展。CFRunLoopSourceContext version0;     /* immutable, except invalidation */ // CFRunLoopSource的原始版本的上下文结构,包含source操作的各种回调函数指针等。CFRunLoopSourceContext1 version1;    /* immutable, except invalidation */ // 更新版本的上下文结构,可能包含更多的回调或信息,也是不可变的,除了失效操作。} _context; // 上下文信息,是source行为的核心,包含了当source被触发时需要执行的操作细节,如处理函数、取消函数等。
};

source0:

```c
/*
定义了一个CFRunLoopSourceContext结构体,这是CFRunLoopSource的核心部分,包含了控制和管理事件源所需的一系列函数指针。
这个结构体允许开发者自定义事件源的行为,当事件源被触发时,RunLoop会调用这些函数执行相应操作。
*/typedef struct {CFIndex version; // 结构体的版本号,用于兼容性检查,通常初始化为0。void *info; // 一个通用指针,开发者可以用来存储任何与事件源相关的自定义数据。const void *(*retain)(const void *info); // 类似于Objective-C的retain方法,用于增加引用计数,保持info对象的生命期。void (*release)(const void *info); // 类似于release方法,用于减少引用计数,当不再需要时帮助释放资源。CFStringRef (*copyDescription)(const void *info); // 返回一个描述info的CFStringRef,通常用于日志和调试。Boolean (*equal)(const void *info1, const void *info2); // 比较两个info指针指向的内容是否相等,用于确定源的唯一性等。CFHashCode (*hash)(const void *info); // 计算info的哈希值,用于快速比较或作为字典键等。void (*schedule)(void *info, CFRunLoopRef rl, CFStringRef mode); // 将事件源添加到指定的RunLoop和运行模式中,使之能够被监控和触发。void (*cancel)(void *info, CFRunLoopRef rl, CFStringRef mode); // 从RunLoop中移除事件源,停止监控和处理。void (*perform)(void *info); // 当事件源被触发时,RunLoop调用此函数执行实际的任务逻辑。
} CFRunLoopSourceContext;

source1:

```c
/*
定义了CFRunLoopSourceContext1结构体,它是CFRunLoopSourceContext的一个扩展版本,专为特定平台提供了额外的功能。
此结构体在基本的CFRunLoopSourceContext基础上,根据目标操作系统条件增加了与mach_port相关的功能,
旨在支持更广泛的系统间通信和同步需求,尤其是在macOS和iOS平台上。
*/typedef struct {CFIndex version; // 结构体版本号,用于版本控制和向前/向后兼容。void *info; // 保留给开发者使用的通用指针,存储与事件源相关的自定义数据。const void *(*retain)(const void *info); // 引用计数管理,类似于retain操作,增加info对象的引用计数。void (*release)(const void *info); // 引用计数管理,减少info对象的引用计数,必要时释放资源。CFStringRef (*copyDescription)(const void *info); // 返回描述info的CFStringRef,常用于日志输出和调试。Boolean (*equal)(const void *info1, const void *info2); // 判断两个info指针所指内容是否相等,用于比较。CFHashCode (*hash)(const void *info); // 计算info的哈希值,用于快速比较或作为字典键。// 下面的函数根据目标操作系统有所不同,提供了mach_port相关的接口或标准的perform调用。#if (TARGET_OS_MAC && !(TARGET_OS_EMBEDDED || TARGET_OS_IPHONE)) || (TARGET_OS_EMBEDDED || TARGET_OS_IPHONE)mach_port_t (*getPort)(void *info); // 获取与事件源关联的mach_port,用于更底层的系统通信和同步。void *  (*perform)(void *msg, CFIndex size, CFAllocatorRef allocator, void *info); // 扩展的perform函数,接受消息、大小、分配器和info,提供更灵活的执行机制。
#elsevoid *  (*getPort)(void *info); // 获取与事件源关联的标准port,虽然命名相同,但此处不特别指mach_port。void    (*perform)(void *info); // 标准的perform函数,直接执行与事件源关联的任务。
#endif
} CFRunLoopSourceContext1;

source0和source1的源码中大概了解引用技术方法和perform方法就行了;

  • source0是非基于Port的。只包含了一个回调(函数指针),它并不能主动触发事件。使用时,你需要先调用
    CFRunLoopSourceSignal(source),将这个 Source 标记为待处理,然后手动调用
    CFRunLoopWakeUp(runloop) 来唤醒 RunLoop,让其处理这个事件。
  • Source1除了包含回调指针外包含一个mach port,Source1可以监听系统端口和通过内核和其他线程通信,接收、分发系统事件,它能够主动唤醒RunLoop(由操作系统内核进行管理

通过了解工作流程,给我的感觉像是source1分析任务并唤醒并分发解析后的任务给source0去处理 ;

CFRunLoopObserver

CFRunLoopObserver是观察者,可以观察RunLoop的各种状态,并抛出回调。结构如下:

struct __CFRunLoopObserver {CFRuntimeBase _base;pthread_mutex_t _lock;CFRunLoopRef _runLoop;CFIndex _rlCount;CFOptionFlags _activities;      /* immutable */CFIndex _order;         /* immutable */CFRunLoopObserverCallBack _callout; /* immutable */CFRunLoopObserverContext _context;  /* immutable, except invalidation */
};

观察状态:

/* Run Loop Observer Activities */
typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {kCFRunLoopEntry = (1UL << 0), //即将进入run loopkCFRunLoopBeforeTimers = (1UL << 1), //即将处理timerkCFRunLoopBeforeSources = (1UL << 2),//即将处理sourcekCFRunLoopBeforeWaiting = (1UL << 5),//即将进入休眠kCFRunLoopAfterWaiting = (1UL << 6),//被唤醒但是还没开始处理事件kCFRunLoopExit = (1UL << 7),//run loop已经退出kCFRunLoopAllActivities = 0x0FFFFFFFU
};

Runloop 通过监控 Source 来决定有没有任务要做,除此之外,我们还可以用 Runloop Observer 来监控 Runloop 本身的状态。 Runloop Observer 可以监控上面的 Runloop 事件

在这里插入图片描述

CFRunLoopTimer

CFRunLoopTimer是定时器,可以在设定的时间点抛出回调
结构如下:

struct __CFRunLoopTimer {CFRuntimeBase _base;uint16_t _bits;  //标记fire状态pthread_mutex_t _lock;CFRunLoopRef _runLoop;        //添加该timer的runloopCFMutableSetRef _rlModes;     //存放所有 包含该timer的 mode的 modeName,意味着一个timer可能会在多个mode中存在CFAbsoluteTime _nextFireDate;CFTimeInterval _interval;     //理想时间间隔  /* immutable */CFTimeInterval _tolerance;    //时间偏差      /* mutable */uint64_t _fireTSR;          /* TSR units */CFIndex _order;         /* immutable */CFRunLoopTimerCallBack _callout;    /* immutable */CFRunLoopTimerContext _context; /* immutable, except invalidation */
};
  • CFRunLoopTimerRef是基于时间的触发器,它和NSTimer可以混用。其包含一个时间长度和一个回调(函数指针)。当其加入到RunLoop时,RunLoop会注册对应的时间点,当时间点到时,RunLoop会被唤醒以执行那个回调。

runloop的实现

实现包括三个点,runloop的获取,添加mode,添加runloopSource ;

runloop的获取

在开放方法或函数上,苹果不允许我们直接创建runloop对象,具体可以使用的创建方法前面也讲到过:

  • CFRunLoopRef CFRunLoopGetCurrent(void)
  • CFRunLoopRef CFRunLoopGetMain(void)
  • +(NSRunLoop *)currentRunLoop
  • +(NSRunLoop *)mainRunLoop

后面两个方法是前面两个方法的封装,现在给出后两个方法的源码:

//取当前所在线程的RunLoop
CFRunLoopRef CFRunLoopGetCurrent(void) {CHECK_FOR_FORK();CFRunLoopRef rl = (CFRunLoopRef)_CFGetTSD(__CFTSDKeyRunLoop);if (rl) return rl;//传入当前线程return _CFRunLoopGet0(pthread_self());
}
//取主线程的RunLoop
CFRunLoopRef CFRunLoopGetMain(void) {CHECK_FOR_FORK();static CFRunLoopRef __main = NULL; // no retain needed//传入主线程if (!__main) __main = _CFRunLoopGet0(pthread_main_thread_np()); // no CAS neededreturn __main;
}

至于内部的CFRunLoopGet0这里就不再给出了 ;

添加Mode

在Core Foundation中,针对Mode的操作,苹果只开放了以下3个API:

  • CFRunLoopAddCommonMode(CFRunLoopRef rl, CFStringRef mode)
  • CFStringRef CFRunLoopCopyCurrentMode(CFRunLoopRef rl)
  • CFArrayRef CFRunLoopCopyAllModes(CFRunLoopRef rl)

我们没有办法直接创建一个CFRunLoopMode对象,但是我们可以调用CFRunLoopAddCommonMode传入一个字符串向RunLoop中添加Mode,传入的字符串即为Mode的名字,Mode对象应该是此时在RunLoop内部创建的。

CFRunLoopAddCommonMode
`CFRunLoopAddCommonMode`函数是Apple的macOS和iOS操作系统中Core Foundation框架的一部分。它用于管理运行循环(run loops),这是一种基本的事件处理循环,负责协调接收和处理系统事件,如用户输入、网络连接、定时器等。下面是该函数的中文注释版本:```c
// 向给定的运行循环中添加一个公共模式。
void CFRunLoopAddCommonMode(CFRunLoopRef rl, CFStringRef modeName) {// 检查是否有进程fork发生,这是一个内部检查,确保运行循环在fork后不会在错误的进程中运行。CHECK_FOR_FORK();// 如果运行循环正在被释放,则直接返回不做任何操作。if (__CFRunLoopIsDeallocating(rl)) return;// 加锁,以确保线程安全。__CFRunLoopLock(rl);// 检查此运行循环是否已包含指定的模式,如果已包含则不执行任何操作。if (!CFSetContainsValue(rl->_commonModes, modeName)) {// 如果已有公共模式项存在,则先复制这些项到一个新的集合中。CFSetRef set = rl->_commonModeItems ? CFSetCreateCopy(kCFAllocatorSystemDefault, rl->_commonModeItems) : NULL;// 将新模式名添加到运行循环的公共模式集合中。CFSetAddValue(rl->_commonModes, modeName);// 如果之前有公共模式项被复制出来,则遍历这些项,并将它们添加到新添加的公共模式下。if (NULL != set) {// 准备一个上下文数组,用于传递给回调函数,包含运行循环引用和模式名称。CFTypeRef context[2] = {rl, modeName};// 遍历并添加项目到新模式下。这会间接调用CFRunLoopAddSource等函数,// 并在需要时创建CFRunLoopMode对象。CFSetApplyFunction(set, (__CFRunLoopAddItemsToCommonMode), (void *)context);// 释放之前复制的集合。CFRelease(set);}} else {// 如果模式已经存在,则什么也不做。}// 解锁,完成操作。__CFRunLoopUnlock(rl);
}

注意的点:1.modeName不能重复,modeName是mode的唯一标识符 2.CFRunLoopMode对象在CFRunLoopAddItemsToCommonMode函数中调用CFRunLoopFindMode时被创建 3.添加commonMode会把commonModeItems数组中的所有source同步到新添加的mode中 4.RunLoop的_commonModes数组存放所有被标记为common的mode的名称

CFRunLoopCopyCurrentMode/CFRunLoopCopyAllModes这两个函数逻辑比较简单,直接取RunLoop的_currentMode和_modes返回 ;

添加Run Loop Source(ModeItem)

  • void CFRunLoopAddSource(CFRunLoopRef rl, CFRunLoopSourceRef source,
    CFStringRef mode)
  • void CFRunLoopRemoveSource(CFRunLoopRef rl, CFRunLoopSourceRef
    source, CFStringRef mode)
  • void CFRunLoopAddObserver(CFRunLoopRef rl, CFRunLoopObserverRef
    observer, CFStringRef mode)
  • void CFRunLoopRemoveObserver(CFRunLoopRef rl, CFRunLoopObserverRef
    observer, CFStringRef * mode)
  • void CFRunLoopAddTimer(CFRunLoopRef rl, CFRunLoopTimerRef timer,
    CFStringRef mode)
  • void CFRunLoopRemoveTimer(CFRunLoopRef rl, CFRunLoopTimerRef timer,
    CFStringRef mode)

runloop运行

这里看别人博客有两种说法,这里展示比较底层的那种
在Core Foundation中我们可以通过以下2个API来让RunLoop运行:

void CFRunLoopRun(void)

在默认的mode下运行当前线程的RunLoop。

CFRunLoopRunResult CFRunLoopRunInMode(CFStringRef mode, CFTimeInterval seconds, Boolean returnAfterSourceHandled)

在指定mode下运行当前线程的RunLoop。

//默认运行runloop的kCFRunLoopDefaultMode
void CFRunLoopRun(void) {   /* DOES CALLOUT */int32_t result;do {//默认在kCFRunLoopDefaultMode下运行runloopresult = CFRunLoopRunSpecific(CFRunLoopGetCurrent(), kCFRunLoopDefaultMode, 1.0e10, false);CHECK_FOR_FORK();} while (kCFRunLoopRunStopped != result && kCFRunLoopRunFinished != result);
}
SInt32 CFRunLoopRunInMode(CFStringRef modeName, CFTimeInterval seconds, Boolean returnAfterSourceHandled) {     /* DOES CALLOUT */CHECK_FOR_FORK();return CFRunLoopRunSpecific(CFRunLoopGetCurrent(), modeName, seconds, returnAfterSourceHandled);
}

这里还可以看出,虽然RunLoop有很多个mode,但是RunLoop在run的时候必须只能指定其中一个mode,运行起来之后,被指定的mode即为currentMode。

这两个函数都调用了一个更加底层的函数:

CFRunLoopRunSpecific

以下是该段代码的中文注释版,描述了如何在指定模式下运行运行循环及其相关流程:```c
/** 在指定模式下运行运行循环* @param rl 当前运行的运行循环引用* @param modeName 需要运行的模式名称* @param seconds 运行循环的超时时间,单位秒* @param returnAfterSourceHandled 处理完事件后是否立即返回* @return 运行结果*/
SInt32 CFRunLoopRunSpecific(CFRunLoopRef rl, CFStringRef modeName, CFTimeInterval seconds, Boolean returnAfterSourceHandled) {// 检查是否有进程fork发生CHECK_FOR_FORK();// 如果运行循环正在被释放,则直接返回运行结束状态if (__CFRunLoopIsDeallocating(rl)) return kCFRunLoopRunFinished;// 加锁以保护运行循环的状态__CFRunLoopLock(rl);// 根据模式名称查找对应的运行模式CFRunLoopModeRef currentMode = __CFRunLoopFindMode(rl, modeName, false);// 如果找不到模式或模式中没有注册任何事件源,则不运行循环直接返回if (NULL == currentMode || __CFRunLoopModeIsEmpty(rl, currentMode, rl->_currentMode)) {Boolean did = false;if (currentMode) __CFRunLoopModeUnlock(currentMode); // 解锁模式(如果已锁定)__CFRunLoopUnlock(rl); // 解锁运行循环return did ? kCFRunLoopRunHandledSource : kCFRunLoopRunFinished;}// 保存本次运行前的数据状态volatile _per_run_data *previousPerRun = __CFRunLoopPushPerRunData(rl);// 记录上一次运行的模式CFRunLoopModeRef previousMode = rl->_currentMode;// 更新当前运行模式rl->_currentMode = currentMode;// 初始化运行结果为“已完成”int32_t result = kCFRunLoopRunFinished;// 1. 通知观察者:运行循环即将进入if (currentMode->_observerMask & kCFRunLoopEntry) __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopEntry);// 2. 实际运行运行循环result = __CFRunLoopRun(rl, currentMode, seconds, returnAfterSourceHandled, previousMode);// 10. 通知观察者:运行循环已退出if (currentMode->_observerMask & kCFRunLoopExit) __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopExit);// 解锁当前模式__CFRunLoopModeUnlock(currentMode);// 恢复之前保存的运行数据状态__CFRunLoopPopPerRunData(rl, previousPerRun);// 还原上一次的运行模式rl->_currentMode = previousMode;// 最终解锁运行循环__CFRunLoopUnlock(rl);// 返回运行结果return result;
}//这段代码详细展示了在给定模式下启动并运行一个运行循环的流程,包括初始化、通知观察者、实际执行循环、以及最终清理和返回结果等关键步骤。

如果指定了一个不存在的mode来运行RunLoop,那么会失败,mode不会被创建,所以这里传入的mode必须是存在的
如果指定了一个mode,但是这个mode中不包含任何modeItem,那么RunLoop也不会运行,所以必须要* 传入至少包含一个modeItem的mode
在进入run loop之前通知observer,状态为kCFRunLoopEntry
在退出run loop之后通知observer,状态为kCFRunLoopExit

__CFRunLoopRun

RunLoop的运行的最核心函数是__CFRunLoopRun,这个函数直接执行了部分工作流程;
源码如下(这里直接用别人公开的源码以及部分注释):

/***  运行run loop**  @param rl              运行的RunLoop对象*  @param rlm             运行的mode*  @param seconds         run loop超时时间*  @param stopAfterHandle true:run loop处理完事件就退出  false:一直运行直到超时或者被手动终止*  @param previousMode    上一次运行的mode**  @return 返回4种状态*/
static int32_t __CFRunLoopRun(CFRunLoopRef rl, CFRunLoopModeRef rlm, CFTimeInterval seconds, Boolean stopAfterHandle, CFRunLoopModeRef previousMode) {//获取系统启动后的CPU运行时间,用于控制超时时间uint64_t startTSR = mach_absolute_time();//如果RunLoop或者mode是stop状态,则直接return,不进入循环if (__CFRunLoopIsStopped(rl)) {__CFRunLoopUnsetStopped(rl);return kCFRunLoopRunStopped;} else if (rlm->_stopped) {rlm->_stopped = false;return kCFRunLoopRunStopped;}//mach端口,在内核中,消息在端口之间传递。 初始为0mach_port_name_t dispatchPort = MACH_PORT_NULL;//判断是否为主线程Boolean libdispatchQSafe = pthread_main_np() && ((HANDLE_DISPATCH_ON_BASE_INVOCATION_ONLY && NULL == previousMode) || (!HANDLE_DISPATCH_ON_BASE_INVOCATION_ONLY && 0 == _CFGetTSD(__CFTSDKeyIsInGCDMainQ)));//如果在主线程 && runloop是主线程的runloop && 该mode是commonMode,则给mach端口赋值为主线程收发消息的端口if (libdispatchQSafe && (CFRunLoopGetMain() == rl) && CFSetContainsValue(rl->_commonModes, rlm->_name)) dispatchPort = _dispatch_get_main_queue_port_4CF();#if USE_DISPATCH_SOURCE_FOR_TIMERSmach_port_name_t modeQueuePort = MACH_PORT_NULL;if (rlm->_queue) {//mode赋值为dispatch端口_dispatch_runloop_root_queue_perform_4CFmodeQueuePort = _dispatch_runloop_root_queue_get_port_4CF(rlm->_queue);if (!modeQueuePort) {CRASH("Unable to get port for run loop mode queue (%d)", -1);}}
#endif//GCD管理的定时器,用于实现runloop超时机制dispatch_source_t timeout_timer = NULL;struct __timeout_context *timeout_context = (struct __timeout_context *)malloc(sizeof(*timeout_context));//立即超时if (seconds <= 0.0) { // instant timeoutseconds = 0.0;timeout_context->termTSR = 0ULL;}//seconds为超时时间,超时时执行__CFRunLoopTimeout函数else if (seconds <= TIMER_INTERVAL_LIMIT) {dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, DISPATCH_QUEUE_OVERCOMMIT);timeout_timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);dispatch_retain(timeout_timer);timeout_context->ds = timeout_timer;timeout_context->rl = (CFRunLoopRef)CFRetain(rl);timeout_context->termTSR = startTSR + __CFTimeIntervalToTSR(seconds);dispatch_set_context(timeout_timer, timeout_context); // source gets ownership of contextdispatch_source_set_event_handler_f(timeout_timer, __CFRunLoopTimeout);dispatch_source_set_cancel_handler_f(timeout_timer, __CFRunLoopTimeoutCancel);uint64_t ns_at = (uint64_t)((__CFTSRToTimeInterval(startTSR) + seconds) * 1000000000ULL);dispatch_source_set_timer(timeout_timer, dispatch_time(1, ns_at), DISPATCH_TIME_FOREVER, 1000ULL);dispatch_resume(timeout_timer);}//永不超时else { // infinite timeoutseconds = 9999999999.0;timeout_context->termTSR = UINT64_MAX;}//标志位默认为trueBoolean didDispatchPortLastTime = true;//记录最后runloop状态,用于returnint32_t retVal = 0;do {//初始化一个存放内核消息的缓冲池uint8_t msg_buffer[3 * 1024];
#if DEPLOYMENT_TARGET_MACOSX || DEPLOYMENT_TARGET_EMBEDDED || DEPLOYMENT_TARGET_EMBEDDED_MINImach_msg_header_t *msg = NULL;mach_port_t livePort = MACH_PORT_NULL;
#elif DEPLOYMENT_TARGET_WINDOWSHANDLE livePort = NULL;Boolean windowsMessageReceived = false;
#endif//取所有需要监听的port__CFPortSet waitSet = rlm->_portSet;//设置RunLoop为可以被唤醒状态__CFRunLoopUnsetIgnoreWakeUps(rl);//2.通知observer,即将触发timer回调,处理timer事件if (rlm->_observerMask & kCFRunLoopBeforeTimers) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeTimers);//3.通知observer,即将触发Source0回调if (rlm->_observerMask & kCFRunLoopBeforeSources) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeSources);//执行加入当前runloop的block__CFRunLoopDoBlocks(rl, rlm);//4.处理source0事件//有事件处理返回true,没有事件返回falseBoolean sourceHandledThisLoop = __CFRunLoopDoSources0(rl, rlm, stopAfterHandle);if (sourceHandledThisLoop) {//执行加入当前runloop的block__CFRunLoopDoBlocks(rl, rlm);}//如果没有Sources0事件处理 并且 没有超时,poll为false//如果有Sources0事件处理 或者 超时,poll都为trueBoolean poll = sourceHandledThisLoop || (0ULL == timeout_context->termTSR);//第一次do..whil循环不会走该分支,因为didDispatchPortLastTime初始化是trueif (MACH_PORT_NULL != dispatchPort && !didDispatchPortLastTime) {
#if DEPLOYMENT_TARGET_MACOSX || DEPLOYMENT_TARGET_EMBEDDED || DEPLOYMENT_TARGET_EMBEDDED_MINI//从缓冲区读取消息msg = (mach_msg_header_t *)msg_buffer;//5.接收dispatchPort端口的消息,(接收source1事件)if (__CFRunLoopServiceMachPort(dispatchPort, &msg, sizeof(msg_buffer), &livePort, 0)) {//如果接收到了消息的话,前往第9步开始处理msggoto handle_msg;}
#elif DEPLOYMENT_TARGET_WINDOWSif (__CFRunLoopWaitForMultipleObjects(NULL, &dispatchPort, 0, 0, &livePort, NULL)) {goto handle_msg;}
#endif}didDispatchPortLastTime = false;//6.通知观察者RunLoop即将进入休眠if (!poll && (rlm->_observerMask & kCFRunLoopBeforeWaiting)) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeWaiting);//设置RunLoop为休眠状态__CFRunLoopSetSleeping(rl);// do not do any user callouts after this point (after notifying of sleeping)// Must push the local-to-this-activation ports in on every loop// iteration, as this mode could be run re-entrantly and we don't// want these ports to get serviced.__CFPortSetInsert(dispatchPort, waitSet);__CFRunLoopModeUnlock(rlm);__CFRunLoopUnlock(rl);#if DEPLOYMENT_TARGET_MACOSX || DEPLOYMENT_TARGET_EMBEDDED || DEPLOYMENT_TARGET_EMBEDDED_MINI
#if USE_DISPATCH_SOURCE_FOR_TIMERS//这里有个内循环,用于接收等待端口的消息//进入此循环后,线程进入休眠,直到收到新消息才跳出该循环,继续执行run loopdo {if (kCFUseCollectableAllocator) {objc_clear_stack(0);memset(msg_buffer, 0, sizeof(msg_buffer));}msg = (mach_msg_header_t *)msg_buffer;//7.接收waitSet端口的消息__CFRunLoopServiceMachPort(waitSet, &msg, sizeof(msg_buffer), &livePort, poll ? 0 : TIMEOUT_INFINITY);//收到消息之后,livePort的值为msg->msgh_local_port,if (modeQueuePort != MACH_PORT_NULL && livePort == modeQueuePort) {// Drain the internal queue. If one of the callout blocks sets the timerFired flag, break out and service the timer.while (_dispatch_runloop_root_queue_perform_4CF(rlm->_queue));if (rlm->_timerFired) {// Leave livePort as the queue port, and service timers belowrlm->_timerFired = false;break;} else {if (msg && msg != (mach_msg_header_t *)msg_buffer) free(msg);}} else {// Go ahead and leave the inner loop.break;}} while (1);
#elseif (kCFUseCollectableAllocator) {objc_clear_stack(0);memset(msg_buffer, 0, sizeof(msg_buffer));}msg = (mach_msg_header_t *)msg_buffer;__CFRunLoopServiceMachPort(waitSet, &msg, sizeof(msg_buffer), &livePort, poll ? 0 : TIMEOUT_INFINITY);
#endif#elif DEPLOYMENT_TARGET_WINDOWS// Here, use the app-supplied message queue mask. They will set this if they are interested in having this run loop receive windows messages.__CFRunLoopWaitForMultipleObjects(waitSet, NULL, poll ? 0 : TIMEOUT_INFINITY, rlm->_msgQMask, &livePort, &windowsMessageReceived);
#endif__CFRunLoopLock(rl);__CFRunLoopModeLock(rlm);// Must remove the local-to-this-activation ports in on every loop// iteration, as this mode could be run re-entrantly and we don't// want these ports to get serviced. Also, we don't want them left// in there if this function returns.__CFPortSetRemove(dispatchPort, waitSet);__CFRunLoopSetIgnoreWakeUps(rl);// user callouts now OK again//取消runloop的休眠状态__CFRunLoopUnsetSleeping(rl);//8.通知观察者runloop被唤醒if (!poll && (rlm->_observerMask & kCFRunLoopAfterWaiting)) __CFRunLoopDoObservers(rl, rlm, kCFRunLoopAfterWaiting);//9.处理收到的消息handle_msg:;__CFRunLoopSetIgnoreWakeUps(rl);#if DEPLOYMENT_TARGET_WINDOWSif (windowsMessageReceived) {// These Win32 APIs cause a callout, so make sure we're unlocked first and relocked after__CFRunLoopModeUnlock(rlm);__CFRunLoopUnlock(rl);if (rlm->_msgPump) {rlm->_msgPump();} else {MSG msg;if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE | PM_NOYIELD)) {TranslateMessage(&msg);DispatchMessage(&msg);}}__CFRunLoopLock(rl);__CFRunLoopModeLock(rlm);sourceHandledThisLoop = true;// To prevent starvation of sources other than the message queue, we check again to see if any other sources need to be serviced// Use 0 for the mask so windows messages are ignored this time. Also use 0 for the timeout, because we're just checking to see if the things are signalled right now -- we will wait on them again later.// NOTE: Ignore the dispatch source (it's not in the wait set anymore) and also don't run the observers here since we are polling.__CFRunLoopSetSleeping(rl);__CFRunLoopModeUnlock(rlm);__CFRunLoopUnlock(rl);__CFRunLoopWaitForMultipleObjects(waitSet, NULL, 0, 0, &livePort, NULL);__CFRunLoopLock(rl);__CFRunLoopModeLock(rlm);__CFRunLoopUnsetSleeping(rl);// If we have a new live port then it will be handled below as normal}#endifif (MACH_PORT_NULL == livePort) {CFRUNLOOP_WAKEUP_FOR_NOTHING();// handle nothing//通过CFRunloopWake唤醒} else if (livePort == rl->_wakeUpPort) {CFRUNLOOP_WAKEUP_FOR_WAKEUP();//什么都不干,跳回2重新循环// do nothing on Mac OS
#if DEPLOYMENT_TARGET_WINDOWS// Always reset the wake up port, or risk spinning foreverResetEvent(rl->_wakeUpPort);
#endif}
#if USE_DISPATCH_SOURCE_FOR_TIMERS//如果是定时器事件else if (modeQueuePort != MACH_PORT_NULL && livePort == modeQueuePort) {CFRUNLOOP_WAKEUP_FOR_TIMER();//9.1 处理timer事件if (!__CFRunLoopDoTimers(rl, rlm, mach_absolute_time())) {// Re-arm the next timer, because we apparently fired early__CFArmNextTimerInMode(rlm, rl);}}
#endif
#if USE_MK_TIMER_TOO//如果是定时器事件else if (rlm->_timerPort != MACH_PORT_NULL && livePort == rlm->_timerPort) {CFRUNLOOP_WAKEUP_FOR_TIMER();// On Windows, we have observed an issue where the timer port is set before the time which we requested it to be set. For example, we set the fire time to be TSR 167646765860, but it is actually observed firing at TSR 167646764145, which is 1715 ticks early. The result is that, when __CFRunLoopDoTimers checks to see if any of the run loop timers should be firing, it appears to be 'too early' for the next timer, and no timers are handled.// In this case, the timer port has been automatically reset (since it was returned from MsgWaitForMultipleObjectsEx), and if we do not re-arm it, then no timers will ever be serviced again unless something adjusts the timer list (e.g. adding or removing timers). The fix for the issue is to reset the timer here if CFRunLoopDoTimers did not handle a timer itself. 9308754//9.1处理timer事件if (!__CFRunLoopDoTimers(rl, rlm, mach_absolute_time())) {// Re-arm the next timer__CFArmNextTimerInMode(rlm, rl);}}
#endif//如果是dispatch到main queue的blockelse if (livePort == dispatchPort) {CFRUNLOOP_WAKEUP_FOR_DISPATCH();__CFRunLoopModeUnlock(rlm);__CFRunLoopUnlock(rl);_CFSetTSD(__CFTSDKeyIsInGCDMainQ, (void *)6, NULL);
#if DEPLOYMENT_TARGET_WINDOWSvoid *msg = 0;
#endif//9.2执行block__CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__(msg);_CFSetTSD(__CFTSDKeyIsInGCDMainQ, (void *)0, NULL);__CFRunLoopLock(rl);__CFRunLoopModeLock(rlm);sourceHandledThisLoop = true;didDispatchPortLastTime = true;} else {CFRUNLOOP_WAKEUP_FOR_SOURCE();// Despite the name, this works for windows handles as wellCFRunLoopSourceRef rls = __CFRunLoopModeFindSourceForMachPort(rl, rlm, livePort);// 有source1事件待处理if (rls) {
#if DEPLOYMENT_TARGET_MACOSX || DEPLOYMENT_TARGET_EMBEDDED || DEPLOYMENT_TARGET_EMBEDDED_MINImach_msg_header_t *reply = NULL;//9.2 处理source1事件sourceHandledThisLoop = __CFRunLoopDoSource1(rl, rlm, rls, msg, msg->msgh_size, &reply) || sourceHandledThisLoop;if (NULL != reply) {(void)mach_msg(reply, MACH_SEND_MSG, reply->msgh_size, 0, MACH_PORT_NULL, 0, MACH_PORT_NULL);CFAllocatorDeallocate(kCFAllocatorSystemDefault, reply);}
#elif DEPLOYMENT_TARGET_WINDOWSsourceHandledThisLoop = __CFRunLoopDoSource1(rl, rlm, rls) || sourceHandledThisLoop;
#endif}}
#if DEPLOYMENT_TARGET_MACOSX || DEPLOYMENT_TARGET_EMBEDDED || DEPLOYMENT_TARGET_EMBEDDED_MINIif (msg && msg != (mach_msg_header_t *)msg_buffer) free(msg);
#endif__CFRunLoopDoBlocks(rl, rlm);if (sourceHandledThisLoop && stopAfterHandle) {//进入run loop时传入的参数,处理完事件就返回retVal = kCFRunLoopRunHandledSource;}else if (timeout_context->termTSR < mach_absolute_time()) {//run loop超时retVal = kCFRunLoopRunTimedOut;}else if (__CFRunLoopIsStopped(rl)) {//run loop被手动终止__CFRunLoopUnsetStopped(rl);retVal = kCFRunLoopRunStopped;}else if (rlm->_stopped) {//mode被终止rlm->_stopped = false;retVal = kCFRunLoopRunStopped;}else if (__CFRunLoopModeIsEmpty(rl, rlm, previousMode)) {//mode中没有要处理的事件retVal = kCFRunLoopRunFinished;}//除了上面这几种情况,都继续循环} while (0 == retVal);if (timeout_timer) {dispatch_source_cancel(timeout_timer);dispatch_release(timeout_timer);} else {free(timeout_context);}return retVal;
}

其中还用到了__CFRunLoopServiceMachPort

__CFRunLoopServiceMachPort

这个函数在run loop中起到了至关重要的作用;

/***  接收指定内核端口的消息**  @param port        接收消息的端口*  @param buffer      消息缓冲区*  @param buffer_size 消息缓冲区大小*  @param livePort    暂且理解为活动的端口,接收消息成功时候值为msg->msgh_local_port,超时时为MACH_PORT_NULL*  @param timeout     超时时间,单位是ms,如果超时,则RunLoop进入休眠状态**  @return 接收消息成功时返回true 其他情况返回false*/
static Boolean __CFRunLoopServiceMachPort(mach_port_name_t port, mach_msg_header_t **buffer, size_t buffer_size, mach_port_t *livePort, mach_msg_timeout_t timeout) {Boolean originalBuffer = true;kern_return_t ret = KERN_SUCCESS;for (;;) {      /* In that sleep of death what nightmares may come ... */mach_msg_header_t *msg = (mach_msg_header_t *)*buffer;msg->msgh_bits = 0;  //消息头的标志位msg->msgh_local_port = port;  //源(发出的消息)或者目标(接收的消息)msg->msgh_remote_port = MACH_PORT_NULL; //目标(发出的消息)或者源(接收的消息)msg->msgh_size = buffer_size;  //消息缓冲区大小,单位是字节msg->msgh_id = 0;  //唯一idif (TIMEOUT_INFINITY == timeout) { CFRUNLOOP_SLEEP(); } else { CFRUNLOOP_POLL(); }//通过mach_msg发送或者接收的消息都是指针,//如果直接发送或者接收消息体,会频繁进行内存复制,损耗性能//所以XNU使用了单一内核的方式来解决该问题,所有内核组件都共享同一个地址空间,因此传递消息时候只需要传递消息的指针ret = mach_msg(msg,MACH_RCV_MSG|MACH_RCV_LARGE|((TIMEOUT_INFINITY != timeout) ? MACH_RCV_TIMEOUT : 0)|MACH_RCV_TRAILER_TYPE(MACH_MSG_TRAILER_FORMAT_0)|MACH_RCV_TRAILER_ELEMENTS(MACH_RCV_TRAILER_AV),0,msg->msgh_size,port,timeout,MACH_PORT_NULL);CFRUNLOOP_WAKEUP(ret);//接收/发送消息成功,给livePort赋值为msgh_local_portif (MACH_MSG_SUCCESS == ret) {*livePort = msg ? msg->msgh_local_port : MACH_PORT_NULL;return true;}//MACH_RCV_TIMEOUT//超出timeout时间没有收到消息,返回MACH_RCV_TIMED_OUT//此时释放缓冲区,把livePort赋值为MACH_PORT_NULLif (MACH_RCV_TIMED_OUT == ret) {if (!originalBuffer) free(msg);*buffer = NULL;*livePort = MACH_PORT_NULL;return false;}//MACH_RCV_LARGE//如果接收缓冲区太小,则将过大的消息放在队列中,并且出错返回MACH_RCV_TOO_LARGE,//这种情况下,只返回消息头,调用者可以分配更多的内存if (MACH_RCV_TOO_LARGE != ret) break;//此处给buffer分配更大内存buffer_size = round_msg(msg->msgh_size + MAX_TRAILER_SIZE);if (originalBuffer) *buffer = NULL;originalBuffer = false;*buffer = realloc(*buffer, buffer_size);}HALT;return false;
}

runloop的应用

线程保活

有时我们会需要经常在一个子线程中执行任务,频繁的创建和销毁线程就会造成很多的开销,这时我们可以通过runloop来控制线程的生命周期。

```objective-c
// 定义属性:一个用于存储线程实例,一个用于控制RunLoop是否停止
@property (nonatomic, strong) NSThread *thread;
@property (nonatomic, assign) BOOL stopped;- (void)viewDidLoad {[super viewDidLoad];// 设置视图背景色为绿色self.view.backgroundColor = [UIColor greenColor];// 创建一个按钮用于触发打印操作UIButton *button = [UIButton buttonWithType:UIButtonTypeRoundedRect];[self.view addSubview:button];[button addTarget:self action:@selector(pressPrint) forControlEvents:UIControlEventTouchUpInside]; // 点击按钮时调用pressPrint方法[button setTitle:@"执行任务" forState:UIControlStateNormal];button.frame = CGRectMake(100, 200, 100, 20); // 设置按钮位置和大小// 创建另一个按钮用于停止RunLoopUIButton *stopButton = [UIButton buttonWithType:UIButtonTypeRoundedRect];[self.view addSubview:stopButton];[stopButton addTarget:self action:@selector(pressStop) forControlEvents:UIControlEventTouchUpInside]; // 点击按钮时调用pressStop方法[stopButton setTitle:@"停止RunLoop" forState:UIControlStateNormal];stopButton.frame = CGRectMake(100, 400, 100, 20); // 设置按钮位置和大小// 初始化停止标志为NOself.stopped = NO;// 使用弱引用来避免循环引用__weak typeof(self) weakSelf = self;// 创建并启动一个新线程,在线程中执行RunLoopself.thread = [[NSThread alloc] initWithBlock:^{NSLog(@"线程---开始");// 向当前RunLoop添加端口,保持RunLoop活跃。使用NSDefaultRunLoopMode模式[[NSRunLoop currentRunLoop] addPort:[[NSPort alloc] init] forMode:NSDefaultRunLoopMode];while (!weakSelf.stopped) {// 运行RunLoop直到指定日期或被显式停止,这里使用一个遥远的未来日期,意味着RunLoop会一直运行直到stopped为YES[[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];}NSLog(@"线程---结束");}];[self.thread start]; // 启动线程
}- (void)pressPrint {// 在子线程中执行print方法[self performSelector:@selector(print) onThread:_thread withObject:nil waitUntilDone:NO];
}// 子线程中执行的任务:打印当前线程信息
- (void)print {NSLog(@"%s, %@", __func__, [NSThread currentThread]);
}- (void)pressStop {// 点击停止按钮时尝试停止子线程的RunLoopif (!_stopped) {// 在子线程上调用stop方法[self performSelector:@selector(stop) onThread:_thread withObject:nil waitUntilDone:YES];}
}// 停止子线程的RunLoop
- (void)stop {// 设置停止标志为YESself.stopped = YES;// 使用CoreFoundation框架的函数直接停止当前RunLoopCFRunLoopStop(CFRunLoopGetCurrent());NSLog(@"%s, %@", __func__, [NSThread currentThread]);// 清除对线程的强引用,允许线程在RunLoop停止后被释放self.thread = nil;
}- (void)dealloc {// 当控制器被释放时打印dealloc日志NSLog(@"%s", __func__);
}//这段代码通过在自定义的子线程中运行RunLoop,实现了按钮控制下的异步任务执行与RunLoop的动态停止功能。它展示了一个典型的手动管理RunLoop的场景,特别是在需要长时间后台处理或监听某些事件时非常有用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/10102.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言 | Leetcode C语言题解之第79题单词搜索

题目&#xff1a; 题解&#xff1a; int directions[4][2] {{0, 1}, {0, -1}, {1, 0}, {-1, 0}};bool check(char** board, int boardSize, int boardColSize, int** visited, int i, int j, char* s, int sSize, int k) {if (board[i][j] ! s[k]) {return false;} else if (…

NSSCTF中的web学习(md5())

目录 MD5的学习 [BJDCTF 2020]easy_md5 [LitCTF 2023]Follow me and hack me [LitCTF 2023]Ping [SWPUCTF 2021 新生赛]easyupload3.0 [NSSCTF 2022 Spring Recruit]babyphp MD5的学习 md5()函数&#xff1a; md5($a)&#xff1a;返回a字符串的散列值 md5($a,TRUE)&…

AWS云优化:实现性能和成本的最佳平衡

随着企业数字化转型的加速&#xff0c;对云计算平台的需求也不断增长。AWS作为云计算行业的领导者之一&#xff0c;提供了广泛的云服务和解决方案&#xff0c;帮助企业实现业务的创新和发展。在AWS云上部署应用程序和服务后&#xff0c;对其进行优化是至关重要的&#xff0c;以…

flutter报错

组件相关 type ‘List’ is not a subtype of type ‘List’ children: CardList.map((item) > Container( 加上 *** < Widget>*** 正常 type ‘(dynamic, dynamic) > Container’ is not a subtype of type ‘(CardType) > Widget’ of ‘f’ children: CardL…

YOLO系列笔记(十四)——Compute Canada计算平台及其常见命令介绍

Compute Canada平台及其常见命令介绍 前言优势使用方法1. 检查模块不带版本号带版本号 2. 加载模块3. 检查模块是否加载成功4. 创建虚拟环境5. 编写作业脚本6. 提交作业7. 监控作业状态8. 查看作业开始预计时间9. 查看作业的详细输出10. 取消作业 注意结语 前言 大家好&#x…

【吃透Java手写】4-Tomcat-简易版

【吃透Java手写】Tomcat-简易版-源码解析 1 准备工作1.1 引入依赖1.2 创建一个Tomcat的启动类 2 线程池技术回顾2.1 线程池的使用流程2.2 线程池的参数2.2.1 任务队列&#xff08;workQueue&#xff09;2.2.2 线程工厂&#xff08;threadFactory&#xff09;2.2.3 拒绝策略&…

表面的相似,本质的不同

韩信与韩王信&#xff0c;两个韩信的结局都是被刘邦所杀&#xff0c;似乎结局类似。但是&#xff0c;略加分析&#xff0c;就会发现其中存在本质的区别。 韩信属于必杀。他的王位是要来的&#xff0c;有居功自傲的本意&#xff0c;功高震主而且毫不避讳。而且年轻&#xff0c;…

Acwing2024蓝桥杯FloodFill

AcWing 687. 扫雷 模拟以下样例(10X10): 把扫雷地图转变为数字记录的地图:地雷记作-1,其余表示8个方向有几个地雷,完成后如下图: 接着搜索所有0联通块(为红色矩形),并且把联通块附近不是地雷的点(红色圆形)全标记为-1,如下图: 而答案就是当前该图中大于0的数的数目之和,再加上…

《数据结构与算法之美》学习笔记一

前言&#xff1a;今天开始学习极客时间的课程《数据结构与算法之美》。为撒要学习这个&#xff1f;因为做力扣题太费劲了&#xff0c;自己的基础太差了&#xff01;所以要学习学习。开一个系列记录一下学习笔记。认真学吧&#xff0c;学有所获才不负韶华&#xff01;之前就学过…

【算法】滑动窗口——找到字符串中所有字母异位词

本节博客是对题目——找到字符串中所有字母异位词的从读题到代码实现以及优化的详细解读&#xff0c;有需要借鉴即可。 目录 1.题目2.滑动窗口 哈希数组3.异位词优化4.总结 1.题目 题目链接&#xff1a;LINK 首先来解释一下什么是异位词&#xff0c;所谓“异位词”&#xf…

show profile

功能 当你执行一个复杂的 SQL 查询时&#xff0c;这个命令可以帮助你了解查询的各个部分花费了多少时间&#xff0c;从而找到可能的性能瓶颈。默认情况下&#xff0c;参数处于关闭状态&#xff0c;并保存最近15次的运行结果 开启 查看是否支持 SHOW VARIABLES LIKE profili…

【XR806开发板试用】试用SWD+Jlink调试

XR806开发板&#xff0c;只能使用编写代码&#xff0c;然后通过UART下载&#xff0c;没法在线debug&#xff0c; 效率会差很多&#xff0c;官方没有提供这一方面的资料。 先查CPU&#xff0c; 官方介绍是arm-china的MC1&#xff0c;通过armv8 Architecture refenence manual资料…

跨境电商行业蓬勃发展,武汉星起航引领卖家孵化新潮流

近年来&#xff0c;我国跨境电商行业在政府的大力扶持下呈现出强劲的发展势头。随着国内制造业结构的加速调整与居民消费需求升级态势的持续凸显&#xff0c;跨境出口规模占比稳步提升&#xff0c;跨境进口规模同样不断扩大&#xff0c;行业市场规模持续增长。在这一背景下&…

QT学习(4)——自定义控件

目录 引出自定义一个控件自定义控件定义方法函数widget窗口调用函数 总结 引出 QT学习&#xff08;4&#xff09;——自定义控件 自定义一个控件 自定义控件定义方法函数 #include "smallwid.h" #include "ui_smallwid.h"SmallWid::SmallWid(QWidget *par…

redis抖动问题导致延迟或者断开的处理方案

目录&#xff1a; 1、使用背景2、redis重试机制3、redis重连机制4、其他一些解决redis抖动问题方案 1、使用背景 客户反馈文件偶现打不开&#xff0c;报错现象是session not exist&#xff0c;最终定位是redis抖动导致的延迟/断开的现象&#xff0c;最终研发团方案是加入redis…

Mapreduce | 案例

根据提供的数据文件【test.log】 数据文件格式&#xff1a;姓名,语文成绩,数学成绩,英语成绩 完成如下2个案例&#xff1a; &#xff08;1&#xff09;求每个学科的平均成绩 &#xff08;2&#xff09;将三门课程中任意一门不及格的学生过滤出来 &#xff08;1&#xff09;求每…

Navicat安装配置(注册码)连接MySQL

下载资源 博主给你打包好了安装包&#xff0c;在网盘里&#xff0c;防止你下载到钓鱼软件 快说谢谢博主&#xff08;然后心甘情愿的点个赞~&#x1f60a;&#xff09; navicatformysql.zip_免费高速下载|百度网盘-分享无限制 (baidu.com) 安装流程 ①下载好压缩包后并解压 ② …

【JavaEE精炼宝库】多线程1(认识线程 | 创建线程 | Thread 类)

目录 一、认识线程 1.1 线程的概念&#xff1a; 1.2 为什么需要线程&#xff1a; 1.3 面试题.谈谈进程和线程的区别&#xff1a; 1.4 Java的线程和操作系统线程的关系&#xff1a; 二、创建线程 2.1 创建线程的5种写法&#xff1a; 2.1.1 写法1.继承 Thread 类&#xf…

【redis】Redis五种常用数据类型和内部编码,以及对String字符串类型的总结

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

C++ | Leetcode C++题解之第80题删除有序数组中的重复项II

题目&#xff1a; 题解&#xff1a; class Solution { public:int removeDuplicates(vector<int>& nums) {int n nums.size();if (n < 2) {return n;}int slow 2, fast 2;while (fast < n) {if (nums[slow - 2] ! nums[fast]) {nums[slow] nums[fast];slo…