iOS-淺談RunLoop

前言:本文簡述RunLoop相關(guān)內(nèi)容,如有錯誤請留言指正态贤。

Q:什么是RunLoop舱呻?

運行循環(huán),在程序運行過程中循環(huán)做一些事情

Q:RunLoop的應(yīng)用抵卫?

  • 定時器(Timer)狮荔、PerformSelector
  • GCD Async Main Queue
  • 事件響應(yīng)、手勢識別介粘、界面刷新
  • 網(wǎng)絡(luò)請求
  • AutoreleasePool

Q:RunLoop什么作用殖氏?

  • 保持程序的持續(xù)運行
  • 處理App中的各種事件(比如觸摸事件、定時器事件等)
  • 節(jié)省CPU資源姻采,提高程序性能:該做事時做事雅采,該休息時休息
  • ......

Q:RunLoop對象?如何獲取婚瓜?

iOS中有2套API來訪問和使用RunLoop:

  • Foundation:NSRunLoop宝鼓,是基于CFRunLoopRef的一層OC包裝
  • Core Foundation:CFRunLoopRef,是開源的https://opensource.apple.com/tarballs/CF/
  • 二者的內(nèi)存地址是不同的巴刻,可以理解為CFRunLoopRef包含NSRunLoop

獲取RunLoop對象:
1.Foundation:

[NSRunLoop currentRunLoop]; // 獲得當(dāng)前線程的RunLoop對象
[NSRunLoop mainRunLoop]; // 獲得主線程的RunLoop對象

2.Core Foundation

CFRunLoopGetCurrent(); // 獲得當(dāng)前線程的RunLoop對象
CFRunLoopGetMain(); // 獲得主線程的RunLoop對象

Q:RunLoop與線程之間關(guān)系愚铡?

  • 每條線程都有唯一的一個與之對應(yīng)的RunLoop對象
  • RunLoop保存在一個全局的Dictionary里,線程作為key胡陪,RunLoop作為value
  • 線程剛創(chuàng)建時并沒有RunLoop對象沥寥,RunLoop會在第一次獲取它時創(chuàng)建
  • RunLoop會在線程結(jié)束時銷毀
  • 主線程的RunLoop已經(jīng)自動獲取(創(chuàng)建)柠座,子線程默認(rèn)沒有開啟RunLoop

RunLoop源碼解讀

CFRunLoop:
struct __CFRunLoop {
    CFRuntimeBase _base;
    pthread_mutex_t _lock;          /* locked for accessing mode list */
    __CFPort _wakeUpPort;           // used for CFRunLoopWakeUp 
    Boolean _unused;
    volatile _per_run_data *_perRunData;              // reset for runs of the run loop
    pthread_t _pthread;//線程
    uint32_t _winthread;
    CFMutableSetRef _commonModes;
    CFMutableSetRef _commonModeItems;
    CFRunLoopModeRef _currentMode;
    CFMutableSetRef _modes;
    struct _block_item *_blocks_head;
    struct _block_item *_blocks_tail;
    CFAbsoluteTime _runTime;
    CFAbsoluteTime _sleepTime;
    CFTypeRef _counterpart;
};

CFRunLoopMode:
struct __CFRunLoopMode {
    CFRuntimeBase _base;
    pthread_mutex_t _lock;  /* must have the run loop locked before locking this */
    CFStringRef _name;
    Boolean _stopped;
    char _padding[3];
    CFMutableSetRef _sources0;
    CFMutableSetRef _sources1;
    CFMutableArrayRef _observers;
    CFMutableArrayRef _timers;
    CFMutableDictionaryRef _portToV1SourceMap;
    __CFPortSet _portSet;
    CFIndex _observerMask;
}

總結(jié)為:
struct __CFRunLoop {
    pthread_t _pthread;
    CFMutableSetRef _commonModes;
    CFMutableSetRef _commonModeItems;
    CFRunLoopModeRef _currentMode;
    CFMutableSetRef _modes;
};

CFRunLoopMode:
struct __CFRunLoopMode {
    CFStringRef _name;
    CFMutableSetRef _sources0;
    CFMutableSetRef _sources1;
    CFMutableArrayRef _observers;
    CFMutableArrayRef _timers;
}

CFRunLoopMode

  • CFRunLoopModeRef代表RunLoop的運行模式
  • 一個RunLoop包含若干個Mode邑雅,每個Mode又包含若干個Source0/Source1/Timer/Observer
  • RunLoop啟動時只能選擇其中一個Mode,作為currentMode
  • 如果需要切換Mode妈经,只能退出當(dāng)前Loop淮野,再重新選擇一個Mode進(jìn)入
  • 不同組的Source0/Source1/Timer/Observer能分隔開來,互不影響
  • 如果Mode里沒有任何Source0/Source1/Timer/Observer吹泡,RunLoop會立馬退出

常見的2種Mode:

  • kCFRunLoopDefaultMode(NSDefaultRunLoopMode):App的默認(rèn)Mode骤星,通常主線程是在這個Mode下運行
  • UITrackingRunLoopMode:界面跟蹤 Mode,用于 ScrollView 追蹤觸摸滑動荞胡,保證界面滑動時不受其他 Mode 影響

Q:Source妈踊、Timer、Observer簡單說明泪漂?

Source0
觸摸事件處理
performSelector:onThread:

Source1
基于Port的線程間通信
系統(tǒng)事件捕捉
Source1捕捉,封裝給Source0處理

Timers
NSTimer
performSelector:withObject:afterDelay:

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

Q:CFRunLoopObserverRef幾種狀態(tài)

如下代碼:

/* Run Loop Observer Activities */
typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
    kCFRunLoopEntry = (1UL << 0),//即將進(jìn)入RunLoop
    kCFRunLoopBeforeTimers = (1UL << 1),//即將處理Timer
    kCFRunLoopBeforeSources = (1UL << 2),//即將處理Source
    kCFRunLoopBeforeWaiting = (1UL << 5),//等待休眠
    kCFRunLoopAfterWaiting = (1UL << 6),//休眠中喚醒
    kCFRunLoopExit = (1UL << 7),//退出RunLoop
    kCFRunLoopAllActivities = 0x0FFFFFFFU
};

Q:如何給RunLoop添加Observer歪泳?

{
 //創(chuàng)建Observer
CFRunLoopObserverRef observer = CFRunLoopObserverCreate(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, observeRunLoopActicities, NULL);
// 添加Observer到RunLoop中
CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
// 釋放
CFRelease(observer);
}

void observeRunLoopActicities(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info)
{
    switch (activity) {
        case kCFRunLoopEntry:
            NSLog(@"kCFRunLoopEntry");
            break;
        case kCFRunLoopBeforeTimers:
            NSLog(@"kCFRunLoopBeforeTimers");
            break;
        case kCFRunLoopBeforeSources:
            NSLog(@"kCFRunLoopBeforeSources");
            break;
        case kCFRunLoopBeforeWaiting:
            NSLog(@"kCFRunLoopBeforeWaiting");
            break;
        case kCFRunLoopAfterWaiting:
            NSLog(@"kCFRunLoopAfterWaiting");
            break;
        case kCFRunLoopExit:
            NSLog(@"kCFRunLoopExit");
            break;
        default:
            break;
    }
}

快捷方法:

// 創(chuàng)建Observer
CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(kCFAllocatorDefault, kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
    switch (activity) {
        case kCFRunLoopEntry: {
            //C/C++中帶有Create和Copy的方法必須釋放
            CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
            NSLog(@"kCFRunLoopEntry - %@", mode);
            CFRelease(mode);
            break;
        }
            
        case kCFRunLoopExit: {
            CFRunLoopMode mode = CFRunLoopCopyCurrentMode(CFRunLoopGetCurrent());
            NSLog(@"kCFRunLoopExit - %@", mode);
            CFRelease(mode);
            break;
        }
            
        default:
            break;
    }
});
// 添加Observer到RunLoop中
CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
// 釋放
CFRelease(observer);

Q:RunLoop的運行邏輯萝勤?

01、通知Observers:進(jìn)入Loop
02呐伞、通知Observers:即將處理Timers
03敌卓、通知Observers:即將處理Sources
04、處理Blocks
05伶氢、處理Source0(可能會再次處理Blocks)
06趟径、如果存在Source1,就跳轉(zhuǎn)到第8步
07癣防、通知Observers:開始休眠(等待消息喚醒)
08蜗巧、通知Observers:結(jié)束休眠(被某個消息喚醒)
  01> 處理Timer
  02> 處理GCD Async To Main Queue
  03> 處理Source1
09、處理Blocks
10蕾盯、根據(jù)前面的執(zhí)行結(jié)果幕屹,決定如何操作
  01> 回到第02步
  02> 退出Loop
11、通知Observers:退出Loop

RunLoop運行邏輯圖

Q:RunLoop休眠原理?

等待消息

  • 沒有消息就讓線程休眠
  • 有消息就喚醒線程
RunLoop休眠原理圖

RunLoop應(yīng)用:線程蓖希活

Q:如何達(dá)到線程泵斐荆活效果?

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

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.thread = [[MJThread alloc] initWithTarget:self selector:@selector(run) object:nil];
    [self.thread start];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    [self performSelector:@selector(test) onThread:self.thread withObject:nil waitUntilDone:NO];
}

// 子線程需要執(zhí)行的任務(wù)
- (void)test{
    NSLog(@"%s %@", __func__, [NSThread currentThread]);
}

// 這個方法的目的:線程彼得簦活
- (void)run {
    NSLog(@"%s %@", __func__, [NSThread currentThread]);
    
    // 往RunLoop里面添加Source\Timer\Observer
    [[NSRunLoop currentRunLoop] addPort:[[NSPort alloc] init] forMode:NSDefaultRunLoopMode];
    [[NSRunLoop currentRunLoop] run];
    NSLog(@"%s ----end----", __func__);
}

上述代碼會引發(fā)的問題:因為線程不會銷毀導(dǎo)致控制器也不會銷毀鸥跟,initWithTarget方法使線程對控制器強引用了。

線程笨活锌雀、控制銷毀

@interface ViewController ()
@property (strong, nonatomic) MJThread *thread;
@property (assign, nonatomic, getter=isStoped) BOOL stopped;
@end

- (void)viewDidLoad {
    [super viewDidLoad];
    
    __weak typeof(self) weakSelf = self;
    
    self.stopped = NO;
    self.thread = [[MJThread alloc] initWithBlock:^{
        NSLog(@"%@----begin----", [NSThread currentThread]);
        
        // 往RunLoop里面添加Source\Timer\Observer
        [[NSRunLoop currentRunLoop] addPort:[[NSPort alloc] init] forMode:NSDefaultRunLoopMode];
        while (!weakSelf.isStoped) {
            [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        }
        
        NSLog(@"%@----end----", [NSThread currentThread]);
        
        // NSRunLoop的run方法是無法停止的,它專門用于開啟一個永不銷毀的線程(NSRunLoop)
        //        [[NSRunLoop currentRunLoop] run];
        /*
         it runs the receiver in the NSDefaultRunLoopMode by repeatedly invoking runMode:beforeDate:.
         In other words, this method effectively begins an infinite loop that processes data from the run loop’s input sources and timers
         */
    }];
    [self.thread start];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    [self performSelector:@selector(test) onThread:self.thread withObject:nil waitUntilDone:NO];
}

// 子線程需要執(zhí)行的任務(wù)
- (void)test{
    NSLog(@"%s %@", __func__, [NSThread currentThread]);
}

- (void)stop {
    // 在子線程調(diào)用stop
    [self performSelector:@selector(stopThread) onThread:self.thread withObject:nil waitUntilDone:NO];
}

// 用于停止子線程的RunLoop
- (void)stopThread{
    // 設(shè)置標(biāo)記為NO
    self.stopped = YES;
    // 停止RunLoop
    CFRunLoopStop(CFRunLoopGetCurrent());
    NSLog(@"%s %@", __func__, [NSThread currentThread]);
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末迅诬,一起剝皮案震驚了整個濱河市腋逆,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌侈贷,老刑警劉巖惩歉,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機删铃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進(jìn)店門危尿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人嘉冒,你說我怎么就攤上這事。” “怎么了亮垫?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長伟骨。 經(jīng)常有香客問我饮潦,道長,這世上最難降的妖魔是什么携狭? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任继蜡,我火速辦了婚禮,結(jié)果婚禮上逛腿,老公的妹妹穿的比我還像新娘稀并。我一直安慰自己,他們只是感情好单默,可當(dāng)我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布碘举。 她就那樣靜靜地躺著,像睡著了一般雕凹。 火紅的嫁衣襯著肌膚如雪殴俱。 梳的紋絲不亂的頭發(fā)上政冻,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天,我揣著相機與錄音线欲,去河邊找鬼明场。 笑死,一個胖子當(dāng)著我的面吹牛李丰,可吹牛的內(nèi)容都是我干的苦锨。 我是一名探鬼主播,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼趴泌,長吁一口氣:“原來是場噩夢啊……” “哼舟舒!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起嗜憔,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤秃励,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后吉捶,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體夺鲜,經(jīng)...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年呐舔,在試婚紗的時候發(fā)現(xiàn)自己被綠了币励。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡珊拼,死狀恐怖食呻,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情澎现,我是刑警寧澤仅胞,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站昔头,受9級特大地震影響饼问,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜揭斧,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望峻堰。 院中可真熱鬧讹开,春花似錦、人聲如沸捐名。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽镶蹋。三九已至成艘,卻和暖如春赏半,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背淆两。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工断箫, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人秋冰。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓仲义,卻偏偏與公主長得像,于是被迫代替她去往敵國和親剑勾。 傳聞我的和親對象是個殘疾皇子埃撵,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,927評論 2 355

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