performSelector: withObject: afterDelay:在線程中執(zhí)行的問題

上一次倉促面試糕簿,給了一份面試題讓我去做疏咐,題目的主要內(nèi)容就是多線程的相關(guān)知識。其中有一題是這樣的:

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"這是1");
        [self performSelector:@selector(test) withObject:nil afterDelay:0];
        NSLog(@"這是3");
    });
}

-(void)test{
    NSLog(@"這是2");
}

讓我寫出程序的打印順序兼蕊。我寫了1,3,2,結(jié)果不用說是錯了砌函,應(yīng)該是1,3。然后回來就立馬去查詢相關(guān)資料讹俊,才知道錯誤原因垦沉。

首先我查看了這個方法的官方API,如圖:
屏幕快照 2019-06-25 下午6.10.40.png

API上面說的很清楚,該方法會在當前線程的runloop中添加定時器仍劈,但是我們使用的是異步執(zhí)行+全局并發(fā)厕倍,就會開啟子線程執(zhí)行block中的任務(wù),這個要是不知道贩疙,那就去補一下GCD相關(guān)知識绑青。

performSelector:withObject:afterDelay: 的底層

  • 不在 NSObject 中,而是在NSRunLoop 類中
  • 帶有 afterDelay 的方法屋群,都是在 NSRunLoop 類中定義的

為何在 主線程就可以調(diào)用 test 方法闸婴,在GCD 中卻不能調(diào)用 test 方法?

  • 這是因為 performSelector:withObject:afterDelay: 類的底層調(diào)用了 NSTimer 定時器芍躏。
  • 定時器是要添加到 runloop 中去的邪乍。
  • 這句話的底層就是 往 runloop 中添加了一個定時器。
  • 主線程 默認有 runloop对竣,所以 在主線程 可以調(diào)用 test方法庇楞。
  • 而子線程中沒有 runloop ,所以 不會調(diào)用 test 方法否纬。如果想要在 GCD 中調(diào)用 test 方法吕晌,需要自己開啟 runloop 。

那么問題來了临燃,怎么讓它執(zhí)行呢睛驳,有幾個方法,下面一一介紹:

方法一:

[self performSelector:@selector(test) withObject:nil afterDelay:0];

既然我們是afterDelay是0秒之后膜廊,那么我們就稍微修改一下乏沸,用跟它很相近的方法:

[self performSelector:@selector(test) withObject:nil];

該方法跟上面方法最大的區(qū)別就是不再使用定時器,而是直接執(zhí)行爪瓜,這樣就不存在Runloop啟動不啟動的問題了蹬跃。修改后如下:

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"這是1");
        [self performSelector:@selector(test) withObject:nil];
        NSLog(@"這是3");
    });
}

-(void)test{
    NSLog(@"這是2");
}

執(zhí)行結(jié)果是:

2019-06-25 18:32:20.066710+0800 GCDTest[19808:1137360] 這是1
2019-06-25 18:32:20.066870+0800 GCDTest[19808:1137360] 這是2
2019-06-25 18:32:20.066973+0800 GCDTest[19808:1137360] 這是3

是不是很方便。

方法二

既然上面說到子線程中Runloop沒有啟動铆铆,那就給它一個Runloop讓它啟動蝶缀。具體實現(xiàn)是:

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"這是1");
        [self performSelector:@selector(test) withObject:nil afterDelay:0];
        NSLog(@"這是3");
        //子線程開啟Runloop,注意該方法要寫在performSelector方法之后才有效
        [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
    });
}

-(void)test{
    //子線程關(guān)閉Runloop
    CFRunLoopStop([NSRunLoop currentRunLoop].getCFRunLoop);
    NSLog(@"這是2");
}

打印結(jié)果是:

2019-06-26 11:28:31.507646+0800 GCDTest[6848:233750] 這是1
2019-06-26 11:28:31.507919+0800 GCDTest[6848:233750] 這是3
2019-06-26 11:28:31.508048+0800 GCDTest[6848:233750] 這是2

如果我們此時在test方法中再執(zhí)行一個gcd方法薄货,就不會執(zhí)行翁都,如:

-(void)test{
    //子線程關(guān)閉Runloop
    CFRunLoopStop([NSRunLoop currentRunLoop].getCFRunLoop);
    NSLog(@"這是2");

    dispatch_sync(dispatch_get_global_queue(0, 0), ^{
        [self performSelector:@selector(test2) withObject:nil afterDelay:1];
    });
}

-(void)test2{
    NSLog(@"不會執(zhí)行");
}

打印結(jié)果還是:

2019-06-26 11:29:26.430061+0800 GCDTest[6864:237962] 這是1
2019-06-26 11:29:26.430325+0800 GCDTest[6864:237962] 這是3
2019-06-26 11:29:26.430482+0800 GCDTest[6864:237962] 這是2

這是因為子線程中的runloop已經(jīng)被關(guān)閉了。

方法三

既然我們說主線程的runloop是默認開啟的菲驴,子線程的沒有開啟荐吵,那么我們就把該方法放到主隊列中執(zhí)行就可以了。代碼如下:

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    dispatch_async(dispatch_get_main_queue(), ^{
        NSLog(@"這是1");
        [self performSelector:@selector(test) withObject:nil afterDelay:0];
        NSLog(@"這是3");
    });
}

-(void)test{
    NSLog(@"這是2");
}

-(void)test2{
    NSLog(@"不會執(zhí)行");
}

執(zhí)行結(jié)果如下:

2019-06-26 11:33:29.554860+0800 GCDTest[6896:247259] 這是1
2019-06-26 11:33:29.555022+0800 GCDTest[6896:247259] 這是3
2019-06-26 11:33:29.555248+0800 GCDTest[6896:247259] 這是2

是不是完美解決了這個問題。
好了先煎,覺得對你有幫助的話記得動手點個贊呦贼涩,關(guān)注我,會給你帶來更多關(guān)于iOS底層的相關(guān)知識薯蝎。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末遥倦,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子占锯,更是在濱河造成了極大的恐慌袒哥,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件消略,死亡現(xiàn)場離奇詭異堡称,居然都是意外死亡,警方通過查閱死者的電腦和手機艺演,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進店門却紧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人胎撤,你說我怎么就攤上這事晓殊。” “怎么了伤提?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵巫俺,是天一觀的道長。 經(jīng)常有香客問我肿男,道長介汹,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任次伶,我火速辦了婚禮痴昧,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘冠王。我一直安慰自己,他們只是感情好舌镶,可當我...
    茶點故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布柱彻。 她就那樣靜靜地躺著,像睡著了一般餐胀。 火紅的嫁衣襯著肌膚如雪哟楷。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天否灾,我揣著相機與錄音卖擅,去河邊找鬼。 笑死,一個胖子當著我的面吹牛惩阶,可吹牛的內(nèi)容都是我干的挎狸。 我是一名探鬼主播,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼断楷,長吁一口氣:“原來是場噩夢啊……” “哼锨匆!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起冬筒,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤恐锣,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后舞痰,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體土榴,經(jīng)...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年响牛,在試婚紗的時候發(fā)現(xiàn)自己被綠了玷禽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,965評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡娃善,死狀恐怖论衍,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情聚磺,我是刑警寧澤坯台,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站瘫寝,受9級特大地震影響蜒蕾,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜焕阿,卻給世界環(huán)境...
    茶點故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一咪啡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧暮屡,春花似錦撤摸、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至莺掠,卻和暖如春衫嵌,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背彻秆。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工楔绞, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留结闸,地道東北人。 一個月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓酒朵,卻偏偏與公主長得像桦锄,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子耻讽,可洞房花燭夜當晚...
    茶點故事閱讀 44,914評論 2 355

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

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴謹 對...
    cosWriter閱讀 11,101評論 1 32
  • Object C中創(chuàng)建線程的方法是什么察纯?如果在主線程中執(zhí)行代碼,方法是什么针肥?如果想延時執(zhí)行代碼饼记、方法又是什么? 1...
    AlanGe閱讀 1,739評論 0 17
  • 1 什么是Runloop Runloop就像他的名字一樣慰枕,是線程中的循環(huán)具则。它用來接收循環(huán)中的各種事件和安排線程工作...
    川少葉閱讀 860評論 1 2
  • 轉(zhuǎn)自bireme,原地址:https://blog.ibireme.com/2015/05/18/runloop/...
    乜_啊_閱讀 1,374評論 0 5
  • 2018年數(shù)字貨幣大熊市掘猿,以區(qū)塊鏈病游、交易套利等衍生出來了很多錢包、持幣生息等項目稠通。近期衬衬,一個跟國際著名品牌OMEG...
    余果777閱讀 478評論 0 0