iOS多線程篇-RunLoop

RunLoop

簡述

1、RunLoop是事件接收和分發機制的一個實現
2、并且它能處理App中的各種事件(比如觸摸事件、定時器事件、Selector事件)
3、以及節省CPU資源,提高程序性能:(該做事時做事,該休息時休息)

如何獲取Runloop對象:

這里的話IOS提供了兩套API來訪問或使用RunLoop
    1、CFRunLoopRef      是在 CoreFoundation 框架內的,它提供了純 C 函數的 API,所有這些 API 都是線程安全的。
    2、NSRunLoop         是基于 CFRunLoopRef 的封裝,提供了面向對象的 API,但是這些 API 不是線程安全的。

CFRunLoopRef的代碼是開源的,你可以在這里CFRunLoopRef源碼下載到整個 CoreFoundation 的源碼

每一個線程對應著一個RunLoop,但是線程在創建的時候是沒有RunLoop的,如果你不去獲取它,它會一直沒有,當然必須你自己的主動去獲取,但是在你線程結束的時候,你所獲取的RunLoop也跟著銷毀了。如果你需要在某個線程對你自己的RunLoop執行一些事件的時候,那么你就的在線程未結束之前進行操作,然而在程序中是具有一個主RunLoop的,它用來管理程序的生死,具體的話是在UIApplicationMain里面執行

//具體顯示
int main(int argc, char * argv[]) {
    @autoreleasepool {
        //程序開始執行  會輸出這段語句
        NSLog(@"------------------");
        //可以看出這里面是一直執行的  相當于一個死循環
        int result = UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
        //程序結束了才會執行  會輸出這段語句,在未結束之前這句話是不會執行的
        NSLog(@"++++++++++++++++++");
        return result;
    }
}

獲取的方式

//獲取的兩種方式

1、這種為CFRunLoopRef中的
    CFRunLoopGetCurrent(); // 獲得當前線程的RunLoop對象
    CFRunLoopGetMain(); // 獲得主線程的RunLoop對象

2、這種為NSRunLoop中的
    [NSRunLoop currentRunLoop]; // 獲得當前線程的RunLoop對象
    [NSRunLoop mainRunLoop]; // 獲得主線程的RunLoop對象

相關類

//相關的五個類

1、CFRunLoopRef
    1、代表一個RunLoop對象
2、CFRunLoopModeRef
    1、代表RunLoop的運行模式
        1、一個RunLoop包含若干個Mode,每個Mode又包含若干個Source/Timer/Observer
        2、每次RunLoop啟動時,只能指定其中一個 Mode,如果需要切換Mode,只能退出Loop,再重新指定一個Mode進入
        3、同一時刻只能進行一種模式
    2、蘋果內部提供了五種模式
        1、kCFRunLoopDefaultMode (NSDefaultRunLoopMode)
            1、App的默認Mode,通常主線程是在這個Mode下運行
        2、UITrackingRunLoopMode
            1、界面跟蹤 Mode,用于 ScrollView 追蹤觸摸滑動,保證界面滑動時不受其他 Mode 影響
        //這個通常用不到
        3、UIInitializationRunLoopMode
            1、在剛啟動 App 時第進入的第一個 Mode,啟動完成后就不再使用
        //這個通常用不到
        4、GSEventReceiveRunLoopMode
            1、接受系統事件的內部 Mode
        5、kCFRunLoopCommonModes
            1、這是一個占位用的Mode,這個的話用語言很難表達,后面會看到實例中會使用到這里,大家仔細體會
3、CFRunLoopSourceRef
    1、用來管理所有事件的事件源,包括自定義的事件,以及系統自帶的事件。
    2、Source有兩個版本:Source0 和 Source1
        1、Source0-----為用戶主動觸發的事件
        2、Source1-----通過內核和其他線程相互發送消息。
4、CFRunLoopTimerRef
    1、基本上說的就是NSTimer,基本用法如下實例標示
5、CFRunLoopObserverRef
    1、用來監聽RunLoop的狀態改變
    2、狀態列表
        kCFRunLoopEntry         = (1UL << 0), // 即將進入Loop
        kCFRunLoopBeforeTimers  = (1UL << 1), // 即將處理 Timer
        kCFRunLoopBeforeSources = (1UL << 2), // 即將處理 Source
        kCFRunLoopBeforeWaiting = (1UL << 5), // 即將進入休眠
        kCFRunLoopAfterWaiting  = (1UL << 6), // 剛從休眠中喚醒
        kCFRunLoopExit          = (1UL << 7), // 即將退出Loop
        kCFRunLoopAllActivities = 0x0FFFFFFFU //所有狀態

補充

一個RunLoop有很多Mode,一個Mode里面有很多得Source/Timer/Observer,但是同一時刻只能進行一種模式。 如圖:

實例

  • CFRunLoopTimerRef

  • 首先在我們的storyboard添加一個text view控件
    然后使用代碼

  • 代碼

    -(void)viewDidLoad {
        [super viewDidLoad];
        //在原來使用time的時候,我們是直接這樣寫的,它是直接添加到RunLoop的DefaultMode模式中去得,如果我們去滑動text view的時候,也就是說我們現在操作的是RunLoop的Tracking,因為在前面我們并沒有把time添加到Tracking中去,那么滑動的時候是不會輸出的,
        //創建time
        NSTimer *time = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(show) userInfo:nil repeats:YES];
        //在這里我們把time添加到Tracking中去,那么操作就會發現,默認的是時候它也會輸出,滑動text view的時候他也會輸出了
        [[NSRunLoop currentRunLoop]addTimer:time forMode:UITrackingRunLoopMode];
    }
    -(void)show{
        NSLog(@"%s",__func__);
    }
    

    把time添加到NSDefaultRunLoopMode模式下

    NSTimer *time = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(show) userInfo:nil repeats:YES];
     //在這里的話可以看到,我們滑動的時候它并沒有輸出,因為我們forMode的為NSDefaultRunLoopMode,也就是通常主線程的這個Mode下運行
    [[NSRunLoop currentRunLoop] addTimer:time forMode:NSDefaultRunLoopMode];
    

    把time添加到UITrackingRunLoopMode模式下

    NSTimer *time = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(show) userInfo:nil repeats:YES];
    //    在這里的話可以看到,我們滑動的時候它才會輸出,因為我們forMode的為UITrackingRunLoopMode,
    [[NSRunLoop currentRunLoop] addTimer:time forMode:UITrackingRunLoopMode];
    

    看了上面兩種是不是有種感覺為什么兩者不能共存,那么下面這種就可以看到它們共存了

    NSTimer *time = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(show) userInfo:nil repeats:YES];
    //    在這里的話可以看到,我們不管是程序啟動還是滑動的時候它都會輸出,因為我們forMode的為kCFRunLoopCommonModes,
    [[NSRunLoop currentRunLoop] addTimer:time forMode:kCFRunLoopCommonModes];
    
  • 補充

    關于定時器的話是有兩種的一個是NSTime,但是它是會受RunLoop的模式所影響的,一個是GCD的定時器,它呢是不受RunLoop的模式所影響的,這里的話留給大家一個引子(GCD的定時器是如何不受RunLoop模式的影響),這個也是公司一般很愛問的一個問題。

  • CFRunLoopObserverRef

  • 代碼

    - (void)viewDidLoad {
        [super viewDidLoad];
        /*
            第一個參數:指定如何給obsever分配存儲空間
            第二個參數:需要監聽的類型/kCFRunLoopAllActivities為全部
            第三個參數:是否每次都監聽
            第四個參數:優先級
            第五個參數:監聽狀態改變之后的回調函數
         */
        CFRunLoopObserverRef obsever = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
    //        kCFRunLoopEntry         = (1UL << 0), // 即將進入Loop
    //        kCFRunLoopBeforeTimers  = (1UL << 1), // 即將處理 Timer
    //        kCFRunLoopBeforeSources = (1UL << 2), // 即將處理 Source
    //        kCFRunLoopBeforeWaiting = (1UL << 5), // 即將進入休眠
    //        kCFRunLoopAfterWaiting  = (1UL << 6), // 剛從休眠中喚醒
    //        kCFRunLoopExit          = (1UL << 7), // 即將退出Loop
    //        kCFRunLoopAllActivities = 0x0FFFFFFFU //所有狀態
            switch (activity) {
                case kCFRunLoopEntry:
                    NSLog(@"即將進入Loop");
                    break;
                case kCFRunLoopBeforeTimers:
                    NSLog(@"即將處理 Timer");
                    break;
                case kCFRunLoopBeforeSources:
                    NSLog(@"即將處理 Source");
                    break;
                case kCFRunLoopBeforeWaiting:
                    NSLog(@"即將進入休眠");
                    break;
                case kCFRunLoopAfterWaiting:
                    NSLog(@"剛從休眠中喚醒");
                    break;
                case kCFRunLoopExit:
                    NSLog(@"即將退出Loop");
                    break;
                default:
                    break;
            }
        });
        //給主線程的RunLoop添加一個觀察者
        /*
         第一個參數:需要給那個RunLoop添加觀察者
         第二個參數:需要添加的observer
         第三個參數:在那種模式下監聽
         */
        CFRunLoopAddObserver(CFRunLoopGetMain(), obsever,kCFRunLoopDefaultMode );
        CFRelease(obsever);
    }
    
  • 補充
    這里的話如果打印出來,是會具有很多time和Source的,因為蘋果內部進行了一系列的調用,那么大家可以明顯的看到,這里是如何監聽RunLoop狀態是如何改變的,最后一定要記得去release,因為ARC無法釋放Core Foundation 框架中的Create、Copy、Release

                   本章到此結束
             歡迎各位碼友隨意轉載并指正
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,505評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,556評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,463評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,009評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,778評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,218評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,281評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,436評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,969評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,795評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,993評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,537評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,229評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,659評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,917評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,687評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,990評論 2 374

推薦閱讀更多精彩內容

  • 基本概念 進程 進程是指在系統中正在運行的一個應用程序,而且每個進程之間是獨立的,它們都運行在其專用且受保護的內存...
    小楓123閱讀 908評論 0 1
  • RunLoop的基本了解 **1 . RunLoop字面的意思 : **運行循環 / 跑圈 **2 . 基本作用 ...
    Mario_ZJ閱讀 523評論 1 3
  • RunLoop NSRunLoop是IOS消息機制的處理模式 主要作用 一條線程對應一個RunLoop,主線程的R...
    大沖哥閱讀 227評論 0 0
  • ======================= 前言 RunLoop 是 iOS 和 OSX 開發中非常基礎的一個...
    i憬銘閱讀 894評論 0 4
  • 認識 Runloop Runloop 就是運行循環,如果沒有 Runloop,程序一運行就會退出,有 Runloo...
    BWLi420閱讀 1,617評論 0 19