詳解iOS多線程GCD問(wèn)題
在iOS所有實(shí)現(xiàn)多線程的方案中,GCD應(yīng)該是最有魅力的,因?yàn)镚CD本身是蘋(píng)果公司為多核的并行運(yùn)算提出的解決方案。GCD在工作時(shí)會(huì)自動(dòng)利用更多的處理器核心,以充分利用更強(qiáng)大的機(jī)器。GCD是Grand Central Dispatch的簡(jiǎn)稱(chēng),它是基于C語(yǔ)言的。如果使用GCD,完全由系統(tǒng)管理線程,我們不需要編寫(xiě)線程代碼。只需定義想要執(zhí)行的任務(wù),然后添加到適當(dāng)?shù)恼{(diào)度隊(duì)列(dispatch queue)。GCD會(huì)負(fù)責(zé)創(chuàng)建線程和調(diào)度你的任務(wù),系統(tǒng)直接提供線程管理
dispatch queue分成以下三種:
1)運(yùn)行在主線程的Main queue,通過(guò)dispatch_get_main_queue獲取。
/*! * @function dispatch_get_main_queue * * @abstract * Returns the default queue that is bound to the main thread. * * @discussion * In order to invoke blocks submitted to the main queue, the application must * call dispatch_main(), NSApplicationMain(), or use a CFRunLoop on the main * thread. * * @result * Returns the main queue. This queue is created automatically on behalf of * the main thread before main() is called. */ __OSX_AVAILABLE_STARTING(__MAC_10_6,__IPHONE_4_0) DISPATCH_EXPORT struct dispatch_queue_s _dispatch_main_q; #define dispatch_get_main_queue() \ DISPATCH_GLOBAL_OBJECT(dispatch_queue_t, _dispatch_main_q)
可以看出,dispatch_get_main_queue也是一種dispatch_queue_t。
2)并行隊(duì)列g(shù)lobal dispatch queue,通過(guò)dispatch_get_global_queue獲取,由系統(tǒng)創(chuàng)建三個(gè)不同優(yōu)先級(jí)的dispatch queue。并行隊(duì)列的執(zhí)行順序與其加入隊(duì)列的順序相同。
3)串行隊(duì)列serial queues一般用于按順序同步訪問(wèn),可創(chuàng)建任意數(shù)量的串行隊(duì)列,各個(gè)串行隊(duì)列之間是并發(fā)的。
當(dāng)想要任務(wù)按照某一個(gè)特定的順序執(zhí)行時(shí),串行隊(duì)列是很有用的。串行隊(duì)列在同一個(gè)時(shí)間只執(zhí)行一個(gè)任務(wù)。我們可以使用串行隊(duì)列代替鎖去保護(hù)共享的數(shù)據(jù)。和鎖不同,一個(gè)串行隊(duì)列可以保證任務(wù)在一個(gè)可預(yù)知的順序下執(zhí)行。
serial queues通過(guò)dispatch_queue_create創(chuàng)建,可以使用函數(shù)dispatch_retain和dispatch_release去增加或者減少引用計(jì)數(shù)。
GCD的用法:
// 后臺(tái)執(zhí)行:
dispatch_async(dispatch_get_global_queue(0, 0), ^{
// something
});
// 主線程執(zhí)行:
dispatch_async(dispatch_get_main_queue(), ^{
// something
});
// 一次性執(zhí)行:
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// code to be executed once
});
// 延遲2秒執(zhí)行:
double delayInSeconds = 2.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC);
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
// code to be executed on the main queue after delay
});
// 自定義dispatch_queue_t
dispatch_queue_t urls_queue = dispatch_queue_create("blog.devtang.com", NULL);
dispatch_async(urls_queue, ^{
// your code
});
dispatch_release(urls_queue);
// 合并匯總結(jié)果
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行執(zhí)行的線程一
});
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行執(zhí)行的線程二
});
dispatch_group_notify(group, dispatch_get_global_queue(0,0), ^{
// 匯總結(jié)果
});
一個(gè)應(yīng)用GCD的例子:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSURL * url = [NSURL URLWithString:@"http://www.baidu.com"];
NSError * error;
NSString * data = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:&error];
if (data != nil) {
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"call back, the data is: %@", data);
});
} else {
NSLog(@"error when download:%@", error);
}
});
GCD的另一個(gè)用處是可以讓程序在后臺(tái)較長(zhǎng)久的運(yùn)行。
在沒(méi)有使用GCD時(shí),當(dāng)app被按home鍵退出后,app僅有最多5秒鐘的時(shí)候做一些保存或清理資源的工作。但是在使用GCD后,app最多有10分鐘的時(shí)間在后臺(tái)長(zhǎng)久運(yùn)行。這個(gè)時(shí)間可以用來(lái)做清理本地緩存,發(fā)送統(tǒng)計(jì)數(shù)據(jù)等工作。
讓程序在后臺(tái)長(zhǎng)久運(yùn)行的示例代碼如下:
// AppDelegate.h文件
@property (assign, nonatomic) UIBackgroundTaskIdentifier backgroundUpdateTask;
// AppDelegate.m文件
- (void)applicationDidEnterBackground:(UIApplication *)application
{
[self beingBackgroundUpdateTask];
// 在這里加上你需要長(zhǎng)久運(yùn)行的代碼
[self endBackgroundUpdateTask];
}
- (void)beingBackgroundUpdateTask
{
self.backgroundUpdateTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
[self endBackgroundUpdateTask];
}];
}
- (void)endBackgroundUpdateTask
{
[[UIApplication sharedApplication] endBackgroundTask: self.backgroundUpdateTask];
self.backgroundUpdateTask = UIBackgroundTaskInvalid;
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- iOS 如何高效的使用多線程
- iOS開(kāi)發(fā)中多線程的安全隱患總結(jié)
- iOS中的多線程如何按設(shè)定順序去執(zhí)行任務(wù)詳解
- IOS多線程編程N(yùn)SThread的使用方法
- iOS 多線程總結(jié)之GCD的使用詳解
- 詳解iOS 多線程 鎖 互斥 同步
- iOS開(kāi)發(fā)網(wǎng)絡(luò)篇—實(shí)現(xiàn)大文件的多線程斷點(diǎn)下載
- IOS 多線程GCD詳解
- iOS多線程介紹
- iOS多線程開(kāi)發(fā)——NSThread淺析
- IOS多線程實(shí)現(xiàn)多圖片下載(一)
- IOS多線程實(shí)現(xiàn)多圖片下載(二)
- IOS開(kāi)發(fā)之多線程N(yùn)SThiread GCD NSOperation Runloop
相關(guān)文章
iOS多控制器實(shí)現(xiàn)帶滑動(dòng)動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了iOS多控制器實(shí)現(xiàn)帶滑動(dòng)動(dòng)畫(huà),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06
IOS 開(kāi)發(fā)之自定義按鈕實(shí)現(xiàn)文字圖片位置隨意定制
這篇文章主要介紹了IOS 開(kāi)發(fā)之自定義按鈕實(shí)現(xiàn)文字圖片位置隨意定制的相關(guān)資料,這里附有實(shí)例代碼及實(shí)現(xiàn)效果圖,需要的朋友可以參考下2016-12-12
iOS利用UITableView設(shè)置全屏分隔線的3種方法總結(jié)
這篇文章主要介紹了關(guān)于iOS利用UITableView設(shè)置全屏分隔線的幾種方法的相關(guān)對(duì)比,分析這三種的各自?xún)?yōu)缺點(diǎn),并且分享了設(shè)置UITableView的單元格分割線離屏幕左右的距離為0的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來(lái)一起看看吧。2017-11-11
ios12中遇到的帶input彈窗的錯(cuò)位問(wèn)題的解決方法
這篇文章主要介紹了ios12中遇到的帶input彈窗的錯(cuò)位問(wèn)題的解決方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05
iOS開(kāi)發(fā)之路--微博“更多”頁(yè)面
本文是IOS開(kāi)發(fā)之路系列文章第五篇,主要講訴了,如何制作微博的更多頁(yè)面,并附上效果圖及源碼,需要的朋友可以參考下,希望能有所幫助2014-08-08
iOS 10撥打系統(tǒng)電話(huà)彈出框延遲出現(xiàn)問(wèn)題的解決
iOS10的到來(lái),帶來(lái)了條幅和鎖屏界面的重新設(shè)計(jì),美觀又好看,再加上抬腕喚醒功能,查看需要的信息確實(shí)更便捷了,還能快捷回復(fù)一些通知,十分輕松,但同樣有問(wèn)題,下面這篇文章主要給大家介紹了關(guān)于iOS 10撥打系統(tǒng)電話(huà)彈出框延遲出現(xiàn)問(wèn)題的解決方法,需要的朋友可以參考下。2017-10-10

