Grand Central Dispatch (GCD)是Apple開發的一個多核編程的解決方法。
一. dispatch queue分類
1. 主線程main queue
????????通過dispatch_get_main_queue()獲取,示例如下:
dispatch_queue_t mainQueue = dispatch_get_main_queue();
2. 并行隊列global queue
????????通過dispatch_get_global_queue(long identifier, unsigned long flags)創建。其中identifier參數:全局并發隊列的優先級標識,由系統創建的不同優先級,例如DISPATCH_QUEUE_PRIORITY_HIGH,DISPATCH_QUEUE_PRIORITY_DEFAULT等,其映射到相應的QOS類;flags參數:保留以供將來使用,傳遞除0以外的任何值都可能導致返回一個空值。示例如下:
dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
說明:并行隊列的執行順序與其加入隊列的順序相同。
3.串行隊列serial queue
????????通過dispatch_queue_create(const char * _Nullable label, dispatch_queue_attr_t _Nullable attr)創建。其中label參數:一個附加到隊列的字符串標簽,可選參數,可能是NULL;attr參數:一般為NULL(DISPATCH_QUEUE_SERIAL = NULL),一個預定義的屬性,例如DISPATCH_QUEUE_SERIAL,DISPATCH_QUEUE_CONCURRENT等,或者dispatch_queue_attr_t類型值。示例如下:
dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue", NULL);
? ? ? ? 串行隊列用于按順序同步訪問,可創建任意數量的串行隊列,各個串行隊列之間是并發的,在MRC下,可以使用函數dispatch_retain和dispatch_release去增加或者減少引用計數。
????????當想要任務按照某一個特定的順序執行時,串行隊列是很有用的。串行隊列在同一個時間只執行一個任務。我們可以使用串行隊列代替鎖去保護共享的數據。和鎖不同的是一個串行隊列可以保證任務在一個可預知的順序下執行。
二. GCD的使用
1. 后臺執行
????????dispatch_async(dispatch_get_global_queue(0, 0), ^{
? ? ????????????// 后臺執行代碼
? ? ? ? });
2. 主線程執行
????????dispatch_async(dispatch_get_main_queue(), ^{
????????????????// 一般用于在后臺執行代碼里面,當后臺執行代碼處理完成時,調用該方法返回主線程執行。
????????});
3. 一次性執行
????????static dispatch_once_t onceToken;
????????dispatch_once(&onceToken, ^{
????????????????// 一般用于單例模式,此處代碼只執行一次
????????});
4. 延時執行
????????dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
????????????????// 延時函數,此處代碼延時執行,delayInSeconds為延時時間值,單位是秒
????????});
5. 自定義隊列
????????dispatch_queue_t urls_queue = dispatch_queue_create("urls_queue", NULL);
????????dispatch_async(urls_queue, ^{
????????????????// 執行代碼
????????});
6. 線程組
????????dispatch_group_t group = dispatch_group_create();
????????dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
????????????????// 并行執行的線程一
????????});
????????dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
????????????????// 并行執行的線程二
????????});
????????dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
????????????????// 匯總結果
????????});
7. 一個應用GCD的例子
????????dispatch_async(dispatch_get_global_queue(0, 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);
????????????????}
????????});
8.?程序在后臺較長久的運行
? ??????GCD的另一個用處是可以讓程序在后臺較長久的運行。
????????在沒有使用GCD時,當app被按home鍵退出后,app僅有最多5秒鐘的時候做一些保存或清理資源的工作。但是在使用GCD后,app最多有10分鐘的時間在后臺長久運行。這個時間可以用來做清理本地緩存,發送統計數據等工作。
????????讓程序在后臺長久運行的示例代碼如下:
// AppDelegate.h文件
@property (assign, nonatomic) UIBackgroundTaskIdentifier backgroundUpdateTask;
// AppDelegate.m文件
- (void)applicationDidEnterBackground:(UIApplication *)application {
????????[self beingBackgroundUpdateTask];
????????// 在這里加上你需要長久運行的代碼? ??
????????[self endBackgroundUpdateTask];
}
- (void)beingBackgroundUpdateTask {
????????self.backgroundUpdateTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
????????????????[self endBackgroundUpdateTask];
????????}];
}
- (void)endBackgroundUpdateTask {
????????[[UIApplication sharedApplication] endBackgroundTask: self.backgroundUpdateTask];
????????self.backgroundUpdateTask = UIBackgroundTaskInvalid;
}