GCD簡單使用

開發(fā)中有時(shí)候可能會(huì)有一些很耗時(shí)的操作,不可能說讓屏幕一直卡著,等操作完成嗡贺,那體驗(yàn)就很尷尬了。所以多線程還是需要來一來的鞍帝。

GCD:

并發(fā)隊(duì)列--同步
NSLog(@"---00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_queue_create("aaa", DISPATCH_QUEUE_CONCURRENT);
    dispatch_sync(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
        
    });
    dispatch_sync(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---aa--%@",[NSThread currentThread]);

看看結(jié)果

并發(fā)隊(duì)列-同步.png

可以看出沒有開啟新線程诫睬,而且是順序執(zhí)行的,需要等待前面的任務(wù)完成后才執(zhí)行后面的任務(wù)帕涌。同步執(zhí)行就是一個(gè)個(gè)來摄凡,所以是同步的話,是什么隊(duì)列蚓曼,執(zhí)行順序沒區(qū)別亲澡。同步和異步?jīng)Q定開不開新線程。

并發(fā)隊(duì)列--異步
NSLog(@"---00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_queue_create("bbb", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---bb--%@",[NSThread currentThread]);

編譯結(jié)果如下:


并發(fā)隊(duì)列-異步.png

并發(fā)隊(duì)列-異步辟躏,開啟了新的線程谷扣,不按順序執(zhí)行。編譯結(jié)果容易讓人以為是主線程的打印完再打印新開的線程捎琐,但是這里并不是一定先打印完---bb 哦会涎,可以在打印---bb前面先睡一下試試。

串行隊(duì)列--同步
NSLog(@"---00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_queue_create("cc", DISPATCH_QUEUE_SERIAL);
    dispatch_sync(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_sync(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---cc--%@",[NSThread currentThread]);

編譯結(jié)果如下:


串行隊(duì)列-同步.png

這就沒什么好說的了瑞凑,沒開新的線程末秃,順序執(zhí)行。

串行隊(duì)列--異步
NSLog(@"---00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_queue_create("dd", DISPATCH_QUEUE_SERIAL);
    dispatch_async(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---dd--%@",[NSThread currentThread]);

編譯結(jié)果如下:


串行隊(duì)列-異步.png

可以看出隊(duì)列中的任務(wù)是順序執(zhí)行的籽御。這里也是一樣练慕,并不是一定先打印完---dd 哦,可以在打印---dd前面先睡一下試試技掏。

主隊(duì)列--同步
NSLog(@"---00000--%@",[NSThread currentThread]);//只打印這句
    dispatch_queue_t queue=dispatch_get_main_queue();
    dispatch_sync(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_sync(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---ee--%@",[NSThread currentThread]);

編譯結(jié)果如下:


主隊(duì)列-同步.png

這種情況就死鎖了铃将,主線程在等同步任務(wù)執(zhí)行完,而同步任務(wù)也在等主線程把這個(gè)同步任務(wù)執(zhí)行完哑梳,互相等待劲阎,死鎖了。

主隊(duì)列--異步
NSLog(@"---h00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_get_main_queue();
    dispatch_async(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---h%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---h%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---h ee--%@",[NSThread currentThread]);

編譯結(jié)果如下:


主隊(duì)列-異步.png

將任務(wù)放到主隊(duì)列鸠真,不會(huì)馬上執(zhí)行悯仙,要等到所有除了我們自己添加到主隊(duì)列的任務(wù)的任務(wù)都執(zhí)行完畢才會(huì)執(zhí)行我們自己添加到主隊(duì)列的任務(wù)。所以---h00000后面一定先打印---h ee 再打印剩下的吠卷。

  • 全局隊(duì)列
    這玩意兒跟并發(fā)隊(duì)列一樣就不說了锡垄。
隊(duì)列組
NSLog(@"---hello");
    dispatch_group_t group=dispatch_group_create();
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
        for (int i=6; i<9; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
        
    });
    NSLog(@"---end");

編譯結(jié)果如下:

隊(duì)列組.png

有時(shí)候是這種需求,分別執(zhí)行兩個(gè)耗時(shí)操作祭隔,等這兩個(gè)耗時(shí)操作執(zhí)行完货岭,再做別的操作。如上圖疾渴,前兩個(gè)異步執(zhí)行完千贯,再執(zhí)行dispatch_group_notify里的任務(wù)。

dispatch_group_enter,dispatch_group_leave

這個(gè)時(shí)候有這樣的情況程奠,如果組隊(duì)列異步操作里也是異步那么dispatch_group_notify會(huì)直接響應(yīng)(可以模仿耗時(shí)操作試試)丈牢。這個(gè)時(shí)候就要用dispatch_group_enter,dispatch_group_leave了。

NSLog(@"---hello");
    dispatch_group_t group=dispatch_group_create();
    
    //dispatch_group_enter(group);
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{

        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            [NSThread sleepForTimeInterval:5.0];
            NSLog(@"---h1--%@",[NSThread currentThread]);
            //dispatch_group_leave(group);
        });
    });
    
    //dispatch_group_enter(group);
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
  
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            [NSThread sleepForTimeInterval:8.0];
            NSLog(@"---h2--%@",[NSThread currentThread]);
        
            //dispatch_group_leave(group);
        });
        
    });
  
    dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
            NSLog(@"---h3--%@",[NSThread currentThread]);
        
    });
    NSLog(@"---h end");

編譯結(jié)果如下:

注釋掉dispatch_group_enter 和dispatch_group_leave.png
打開注釋后dispatch_group_enter 和dispatch_group_leave.png

這樣就能保證先執(zhí)行前面的異步任務(wù)之后再執(zhí)行dispatch_group_notify里的任務(wù)瞄沙;

dispatch_group_wait --(在group上任務(wù)完成前己沛,dispatch_group_wait會(huì)阻塞當(dāng)前線程(所以不能放在主線程調(diào)用)一直等待;當(dāng)group上任務(wù)完成距境,或者等待時(shí)間超過設(shè)置的超時(shí)時(shí)間會(huì)結(jié)束等待申尼;)
NSLog(@"---hello");
    dispatch_group_t group=dispatch_group_create();
    
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        
            NSLog(@"---h1--%@",[NSThread currentThread]);
        });
    
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        
            [NSThread sleepForTimeInterval:8.0];
            NSLog(@"---h2--%@",[NSThread currentThread]);
            
        });
    
    dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
        NSLog(@"---h3--%@",[NSThread currentThread]);
        
    });
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5*NSEC_PER_SEC)));
        NSLog(@"---h4---%@",[NSThread currentThread]);
    });
    NSLog(@"---h end");

編譯結(jié)果如下:


超過5秒超時(shí)時(shí)間后走---h4 再走group里的---h2.png
dispatch_barrier_async(柵欄方法(將兩組異步分割開來))
NSLog(@"---00000--%@",[NSThread currentThread]);
    dispatch_queue_t queue=dispatch_queue_create("hehe", DISPATCH_QUEUE_CONCURRENT);
    dispatch_async(queue, ^{
        for (int i=0; i<3; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=3; i<6; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_barrier_async(queue, ^{
        for (int i=6; i<9; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=9; i<12; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    dispatch_async(queue, ^{
        for (int i=12; i<15; i++) {
            NSLog(@"---%d--%@",i,[NSThread currentThread]);
        }
    });
    NSLog(@"---ff--%@",[NSThread currentThread]);

編譯結(jié)果如下:

dispatch_barrier_async.png

可以看出前面兩個(gè)異步先執(zhí)行完,再執(zhí)行的后面兩個(gè)異步垫桂。

延時(shí)
NSLog(@"---hello");
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        //2s后異步執(zhí)行
        NSLog(@"---h---come !");
    });
    NSLog(@"---h end");

編譯結(jié)果如下:

延時(shí)2秒.png
  • dispatch_once單例 就不寫了师幕。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子霹粥,更是在濱河造成了極大的恐慌灭将,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,039評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件后控,死亡現(xiàn)場離奇詭異庙曙,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)浩淘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門捌朴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人张抄,你說我怎么就攤上這事砂蔽。” “怎么了署惯?”我有些...
    開封第一講書人閱讀 165,417評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵左驾,是天一觀的道長。 經(jīng)常有香客問我泽台,道長什荣,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,868評(píng)論 1 295
  • 正文 為了忘掉前任怀酷,我火速辦了婚禮稻爬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蜕依。我一直安慰自己桅锄,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,892評(píng)論 6 392
  • 文/花漫 我一把揭開白布样眠。 她就那樣靜靜地躺著友瘤,像睡著了一般。 火紅的嫁衣襯著肌膚如雪檐束。 梳的紋絲不亂的頭發(fā)上辫秧,一...
    開封第一講書人閱讀 51,692評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音被丧,去河邊找鬼盟戏。 笑死,一個(gè)胖子當(dāng)著我的面吹牛甥桂,可吹牛的內(nèi)容都是我干的柿究。 我是一名探鬼主播,決...
    沈念sama閱讀 40,416評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼黄选,長吁一口氣:“原來是場噩夢啊……” “哼蝇摸!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,326評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤貌夕,失蹤者是張志新(化名)和其女友劉穎律歼,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蜂嗽,經(jīng)...
    沈念sama閱讀 45,782評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡苗膝,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,957評(píng)論 3 337
  • 正文 我和宋清朗相戀三年殃恒,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了植旧。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,102評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡离唐,死狀恐怖病附,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情亥鬓,我是刑警寧澤完沪,帶...
    沈念sama閱讀 35,790評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站嵌戈,受9級(jí)特大地震影響覆积,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜熟呛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,442評(píng)論 3 331
  • 文/蒙蒙 一宽档、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧庵朝,春花似錦吗冤、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,996評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至侄旬,卻和暖如春肺蔚,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背儡羔。 一陣腳步聲響...
    開封第一講書人閱讀 33,113評(píng)論 1 272
  • 我被黑心中介騙來泰國打工宣羊, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人笔链。 一個(gè)月前我還...
    沈念sama閱讀 48,332評(píng)論 3 373
  • 正文 我出身青樓段只,卻偏偏與公主長得像,于是被迫代替她去往敵國和親鉴扫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子赞枕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,044評(píng)論 2 355

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