#Runloop

什么是runloop

運行循環(huán)

應(yīng)用范疇
  • 定時器(Timer)、PerformSelector
  • GCD Async Main Queue
  • 事件響應(yīng)饿自、手勢識別汰翠、界面刷新
  • 網(wǎng)絡(luò)請求
  • AutoreleasePool
RunLoop的基本作用
  • 保持程序的持續(xù)運行
  • 處理App中的各種事件(比如觸摸事件、定時器事件等)
  • 節(jié)省CPU資源昭雌,提高程序性能:該做事時做事复唤,該休息時休息
  • 程序不會立馬退出,處理APP里事件城豁,節(jié)省cpu資源
RunLoop對象
  • iOS中有2套API來訪問和使用RunLoop
    • Foundation:NSRunLoop
      NSRunLoop *runloop = [NSRunLoop currentRunLoop];
    • Core Foundation:CFRunLoopRef
      CFRunLoopRef runloop = CFRunLoopGetCurrent();
      NSRunLoop和CFRunLoopRef都代表著RunLoop對象
      NSRunLoop是基于CFRunLoopRef的一層OC包裝
      CFRunLoopRef是開源的
RunLoop與線程
  • 每條線程都有唯一的一個與之對應(yīng)的RunLoop對象
  • RunLoop保存在一個全局的Dictionary里苟穆,線程作為key抄课,RunLoop作為value
  • 線程剛創(chuàng)建時并沒有RunLoop對象唱星,RunLoop會在第一次獲取它時創(chuàng)建
  • RunLoop會在線程結(jié)束時銷毀
  • 主線程的RunLoop已經(jīng)自動獲取(創(chuàng)建)跟磨,子線程默認沒有開啟RunLoop
獲取runloop對象.png
線程為key,runloop為value.png
RunLoop相關(guān)的類
  • CFRunLoopRef
  • CFRunLoopModeRef
  • CFRunLoopSourceRef
  • CFRunLoopTimerRef
  • CFRunLoopObserverRef
CFRunLoopMode.jpg
CFRunLoopRef.jpg
modes.png
  • Source0
    觸摸事件處理
    performSelector:onThread:

  • Source1
    基于Port的線程間通
    系統(tǒng)捕捉事件(通過source1捕捉 间聊,分發(fā)source0 處理)

  • Timers
    NSTimer
    performSelector:withObject:afterDelay:

  • Observers
    用于監(jiān)聽RunLoop的狀態(tài)
    UI刷新(BeforeWaiting)
    Autorelease pool(BeforeWaiting)

RunLoop 循環(huán)處理不同模式下的 Source0 Source1 Timers Observers

CFRunLoopModeRef
  • 一個runloop有多個模式,只能選擇一種模型運行
  • 一個 RunLoop包含若干個 Mode抵拘,每個Mode又包含若干個Source/Timer/Observer
  • RunLoop啟動時哎榴,只能指定其中一個 Mode,這個Mode被稱作 CurrentMode
  • 如果需要切換Mode僵蛛,只能退出Loop尚蝌,再重新指定一個Mode進入
    (不同組的Source/Timer/Observer,互不影響)
  • 如果Mode里沒有任何Source0/Source1/Timer/Observer充尉,RunLoop會立馬退出

常見Mode

 //App的默認Mode飘言,通常主線程是在這個Mode下運行
UITrackingRunLoopMode
//界面跟蹤 Mode,用于 ScrollView 追蹤觸摸滑動驼侠,保證界面滑動時不受其他 Mode 影響
NSDefaultRunLoopMode

CFRunLoopTimerRef

 NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
// 定時器只運行在NSDefaultRunLoopMode下姿鸿,一旦RunLoop進入其他模式,這個定時器就不會工作
 [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

    // 定時器會跑在標記為common modes的模式下
    // 標記為common modes的模式:UITrackingRunLoopMode和NSDefaultRunLoopMode
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

CFRunLoopSourceRef
事件源(輸入源) 基于理論劃分
Port-Based Sources 基于端口 和其他線程交互 Mac 內(nèi)核
Custom Input Sources 自定義輸入源
Cocoa Perform Selector Sources

函數(shù)調(diào)用棧 (基于實踐劃分)
Source0:非基于Port的

Source1:基于Port的倒源,通過內(nèi)核和其他線程通信苛预,接受、分發(fā)系統(tǒng)事件

CFRunLoopObserverRef

CFRunLoopObserverRef是觀察者笋熬,能夠監(jiān)聽RunLoop的狀態(tài)改變


CFRunLoopObserverRef.png
RunLoop運行邏輯
官方.png
runloop實現(xiàn)邏輯.png
xcode調(diào)用棧.png

Runloop代碼實現(xiàn)


/* rl, rlm are locked on entrance and exit */
static int32_t __CFRunLoopRun(CFRunLoopRef rl, CFRunLoopModeRef rlm, CFTimeInterval seconds, Boolean stopAfterHandle, CFRunLoopModeRef previousMode) {
    
    int32_t retVal = 0;
    do {
        //通知Observers:處理timer
        __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeTimers);
        //通知Observers:處理sources
        __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeSources);
        //處理blocks
        __CFRunLoopDoBlocks(rl, rlm);
        //處理source0
        if (__CFRunLoopDoSources0(rl, rlm, stopAfterHandle)) {
            //處理blocks
            __CFRunLoopDoBlocks(rl, rlm);
        }
        
        //判斷是否有source1
        if (__CFRunLoopServiceMachPort(dispatchPort, &msg, sizeof(msg_buffer), &livePort, 0, &voucherState, NULL)) {
            //如果有source1 ,跳轉(zhuǎn)到 handle_msg
            goto handle_msg;
        }
        //通知Observers:即將休眠
        __CFRunLoopDoObservers(rl, rlm, kCFRunLoopBeforeWaiting);
        __CFRunLoopSetSleeping(rl);
        
        
        __CFRunLoopModeUnlock(rlm);
        __CFRunLoopUnlock(rl);
        //等待別的消息來喚醒當(dāng)前線程
        __CFRunLoopServiceMachPort(waitSet, &msg, sizeof(msg_buffer), &livePort, poll ? 0 : TIMEOUT_INFINITY, &voucherState, &voucherCopy);
        
        __CFRunLoopUnsetSleeping(rl);
        //通知Observers:結(jié)束休眠
        __CFRunLoopDoObservers(rl, rlm, kCFRunLoopAfterWaiting);
        
    handle_msg:;
    if (被timer喚醒) {
        //處理timers
        __CFRunLoopDoTimers(rl, rlm, mach_absolute_time()
        } else if (被GCD喚醒)
        {
            CFRUNLOOP_WAKEUP_FOR_DISPATCH();
        } else { //被source1喚醒
            __CFRunLoopDoSource1(rl, rlm, rls, msg, msg->msgh_size, &reply) ||sourceHandledThisLoop;
        }
        
        //處理blocks
        __CFRunLoopDoBlocks(rl, rlm);
        
        //設(shè)置返回值
        if (sourceHandledThisLoop && stopAfterHandle) {
            retVal = kCFRunLoopRunHandledSource;
        } else if (timeout_context->termTSR < mach_absolute_time()) {
            retVal = kCFRunLoopRunTimedOut;
        } else if (__CFRunLoopIsStopped(rl)) {
            __CFRunLoopUnsetStopped(rl);
            retVal = kCFRunLoopRunStopped;
        } else if (rlm->_stopped) {
            rlm->_stopped = false;
            retVal = kCFRunLoopRunStopped;
        } else if (__CFRunLoopModeIsEmpty(rl, rlm, previousMode)) {
            retVal = kCFRunLoopRunFinished;
        }
        
        } while (0 == retVal);
        
        if (timeout_timer) {
            dispatch_source_cancel(timeout_timer);
            dispatch_release(timeout_timer);
        } else {
            free(timeout_context);
        }
        
        return retVal;
                            
        }
            
                            
SInt32 CFRunLoopRunSpecific(CFRunLoopRef rl, CFStringRef modeName, CFTimeInterval seconds, Boolean returnAfterSourceHandled) {     /* DOES CALLOUT */
    
    //通知Observers:進入Runloop
    __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopEntry);
    //具體要做的事情
    result = __CFRunLoopRun(rl, currentMode, seconds, returnAfterSourceHandled, previousMode);
    //通知Observers:退出Runloop
    __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopExit);
    
    return result;
}


runloop 開發(fā)中應(yīng)用
  • 控制線程生命周期(線程比饶常活)
#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface TGPermenantThread : NSObject

/**
 在當(dāng)前子線程執(zhí)行一個任務(wù)
 */
- (void)executeTask:(void(^)(void))task;

/**
 結(jié)束線程
 */
- (void)stop;

@end

NS_ASSUME_NONNULL_END

#import "TGPermenantThread.h"

/** 使用TGThread是為了調(diào)試,可以使用NSThread替換 **/
@interface TGThread : NSThread
@end
@implementation TGThread
- (void)dealloc
{
    NSLog(@"%s", __func__);
}
@end

/** TGPermenantThread **/
@interface TGPermenantThread()
@property (strong, nonatomic) TGThread *innerThread;
@end

@implementation TGPermenantThread
#pragma mark - public methods
- (instancetype)init
{
    if (self = [super init]) {
        self.innerThread = [[TGThread alloc] initWithBlock:^{
            NSLog(@"begin----");
            
            // 創(chuàng)建上下文(要初始化一下結(jié)構(gòu)體)
            CFRunLoopSourceContext context = {0};
            
            // 創(chuàng)建source
            CFRunLoopSourceRef source = CFRunLoopSourceCreate(kCFAllocatorDefault, 0, &context);
            
            // 往Runloop中添加source
            CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);
            
            // 銷毀source
            CFRelease(source);
            
            // 啟動
            CFRunLoopRunInMode(kCFRunLoopDefaultMode, 1.0e10, false);
        }];
        
        [self.innerThread start];
    }
    return self;
}

- (void)executeTask:(void(^)(void))task
{
    if (!self.innerThread || !task) return;
    
    [self performSelector:@selector(__executeTask:) onThread:self.innerThread withObject:task waitUntilDone:NO];
}

- (void)stop
{
    if (!self.innerThread) return;
    [self performSelector:@selector(__stop) onThread:self.innerThread withObject:nil waitUntilDone:YES];
}

- (void)dealloc
{
    NSLog(@"%s", __func__);
    [self stop];
}

#pragma mark - private methods
- (void)__stop
{
    CFRunLoopStop(CFRunLoopGetCurrent());
    self.innerThread = nil;
}

- (void)__executeTask:(void(^)(void))task{
     task();
}
@end


#import "ViewController.h"
#import "TGPermenantThread.h"

@interface ViewController ()
@property (strong, nonatomic) TGPermenantThread *thread;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.thread = [[TGPermenantThread alloc] init];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self.thread executeTask:^{
        NSLog(@"執(zhí)行任務(wù) - %@", [NSThread currentThread]);
    }];
}

- (IBAction)stop {
    [self.thread stop];
}

- (void)dealloc
{
    NSLog(@"%s", __func__);
}

@end


  • 解決NSTimer在滑動時停止工作的問題
 NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];
// 定時器只運行在NSDefaultRunLoopMode下,一旦RunLoop進入其他模式胳螟,這個定時器就不會工作
 [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

// 定時器會跑在標記為common modes的模式下
// 標記為common modes的模式:UITrackingRunLoopMode和NSDefaultRunLoopMode
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

CFRunLoopSourceRef
事件源(輸入源) 基于理論劃分
Port-Based Sources 基于端口 和其他線程交互 Mac 內(nèi)核
Custom Input Sources 自定義輸入源
Cocoa Perform Selector Sources

函數(shù)調(diào)用棧 (基于實踐劃分)
Source0:非基于Port的

Source1:基于Port的昔馋,通過內(nèi)核和其他線程通信,接受旺隙、分發(fā)系統(tǒng)事件

CFRunLoopObserverRef

CFRunLoopObserverRef是觀察者绒极,能夠監(jiān)聽RunLoop的狀態(tài)改變

添加observer 監(jiān)聽 RunLoop

 
// 創(chuàng)建Observer
    CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
        switch (activity) {
           case kCFRunLoopEntry:
          {
       CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
                NSLog(@"kCFRunLoopEntry - %@", mode);
                CFRelease(mode);
       }
            break;
        case kCFRunLoopBeforeTimers:
            NSLog(@"kCFRunLoopBeforeTimers");
            break;
        case kCFRunLoopBeforeSources:
            NSLog(@"kCFRunLoopBeforeSources");
            break;
        case kCFRunLoopBeforeWaiting:
            NSLog(@"kCFRunLoopBeforeWaiting");
            break;
        case kCFRunLoopAfterWaiting:
            NSLog(@"kCFRunLoopAfterWaiting");
            break;
        case kCFRunLoopExit:
         {
              CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
                NSLog(@"kCFRunLoopExit - %@", mode);
                CFRelease(mode);
        }
            break;
        default:
            break;
        }
    });
    // 添加Observer到RunLoop中
    CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
    // 釋放
    CFRelease(observer);
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市蔬捷,隨后出現(xiàn)的幾起案子垄提,更是在濱河造成了極大的恐慌榔袋,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件铡俐,死亡現(xiàn)場離奇詭異凰兑,居然都是意外死亡,警方通過查閱死者的電腦和手機审丘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進店門吏够,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人滩报,你說我怎么就攤上這事锅知。” “怎么了脓钾?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵售睹,是天一觀的道長。 經(jīng)常有香客問我可训,道長昌妹,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任握截,我火速辦了婚禮飞崖,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘谨胞。我一直安慰自己固歪,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布畜眨。 她就那樣靜靜地躺著昼牛,像睡著了一般。 火紅的嫁衣襯著肌膚如雪康聂。 梳的紋絲不亂的頭發(fā)上贰健,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天,我揣著相機與錄音恬汁,去河邊找鬼伶椿。 笑死,一個胖子當(dāng)著我的面吹牛氓侧,可吹牛的內(nèi)容都是我干的脊另。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼约巷,長吁一口氣:“原來是場噩夢啊……” “哼偎痛!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起独郎,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤踩麦,失蹤者是張志新(化名)和其女友劉穎枚赡,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體谓谦,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡贫橙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了反粥。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片卢肃。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖才顿,靈堂內(nèi)的尸體忽然破棺而出莫湘,到底是詐尸還是另有隱情,我是刑警寧澤娜膘,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布逊脯,位于F島的核電站优质,受9級特大地震影響竣贪,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜巩螃,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一演怎、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧避乏,春花似錦爷耀、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至铆帽,卻和暖如春咆耿,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背爹橱。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工萨螺, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人愧驱。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓慰技,卻偏偏與公主長得像,于是被迫代替她去往敵國和親组砚。 傳聞我的和親對象是個殘疾皇子吻商,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容