以上這三種編程方式從上到下,抽象度層次是從低到高的,抽象度越高的使用越簡單,也是Apple最推薦使用的,在項目中不少框架技術分別使用了不一樣多線程技術。程序員
1> 類方法直接開啓後臺線程,並執行選擇器方法編程
detachNewThreadSelector網絡
1 // 新建一個線程,調用@selector方法 2 3 [NSThread detachNewThreadSelector:@selector(bigDemo) toTarget:self withObject:nil];
2> 成員方法,在實例化線程對象以後,須要使用start執行選擇器方法多線程
initWithTarget併發
// 成員方法 NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(bigDemo) object:nil]; // 啓動start線程 [thread start];
對於NSThread的簡單使用,能夠用NSObject的performSelectorInBackground替代框架
1 // performSelectorInBackground是將bigDemo的任務放在後臺線程中執行 2 3 [self performSelectorInBackground:@selector(bigDemo) withObject:nil];
同時,在NSThread調用的方法中,一樣要使用autoreleasepool進行內存管理,不然容易出現內存泄露。異步
1 // 自動釋放池 2 3 // 負責其餘線程上的內存管理,在使用NSThread或者NSObject的線程方法時,必定要使用自動釋放池 4 5 // 不然容易出現內存泄露。 6 7 @autoreleasepool { 8 9 10 11 }
1> 使用步驟:async
1) 實例化操做spa
1 // 實例化操做隊列 2 _queue = [[NSOperationQueue alloc] init];
a) NSInvocationOperation線程
1 NSInvocationOperation *op1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(opAction) object:nil]; 2 3 // 若是使用start,會在當前線程啓動操做 4 // [op1 start]; 5 6 // 1. 一旦將操做添加到操做隊列,操做就會啓動 7 [_queue addOperation:op1];
b) NSBlockOperation
1 #pragma mark 模仿下載網絡圖像 2 - (IBAction)operationDemo3:(id)sender 3 { 4 // 1. 下載 5 NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{ 6 NSLog(@"下載 %@" , [NSThread currentThread]); 7 }]; 8 // 2. 濾鏡 9 NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{ 10 NSLog(@"濾鏡 %@" , [NSThread currentThread]); 11 }]; 12 // 3. 顯示 13 NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{ 14 NSLog(@"更新UI %@" , [NSThread currentThread]); 15 }]; 16 17 // 添加操做之間的依賴關係,所謂「依賴」關係,就是等待前一個任務完成後,後一個任務才能啓動 18 // 依賴關係能夠跨線程隊列實現 19 // 提示:在指定依賴關係時,注意不要循環依賴,不然不工做。 20 [op2 addDependency:op1]; 21 [op3 addDependency:op2]; 22 // [op1 addDependency:op3]; 23 24 [_queue addOperation:op1]; 25 [_queue addOperation:op2]; 26 [[NSOperationQueue mainQueue] addOperation:op3]; 27 }
2) 將操做添加到隊列NSOperationQueue便可啓動多線程執行
1 [_queue addOperation:op1]; 2 [_queue addOperation:op2];
2> 更新UI使用主線程隊列
//兩方式 [NSOpeationQueue mainQueue] addOperation ^{ }; [[NSOperationQueue mainQueue] addOperation:op3];
3> 操做隊列的setMaxConcurrentOperationCount
能夠設置同時併發的線程數量!
1 // 控制同時最大併發的線程數量 2 [_queue setMaxConcurrentOperationCount:2];
提示:此功能僅有NSOperation有!
4> 使用addDependency能夠設置任務的執行前後順序,同時能夠跨操做隊列指定依賴關係
1 // 添加操做之間的依賴關係,所謂「依賴」關係,就是等待前一個任務完成後,後一個任務才能啓動 2 3 // 依賴關係能夠跨線程隊列實現 4 5 // 提示:在指定依賴關係時,注意不要循環依賴,不然不工做。 6 [op2 addDependency:op1]; 7 [op3 addDependency:op2]; 8 [op1 addDependency:op3];
提示:在指定依賴關係時,注意不要循環依賴,不然不工做。
GCD就是爲了在「多核」上使用多線程技術
1> 要使用GCD,全部的方法都是dispatch開頭的
2> 名詞解釋
global 全局
queue 隊列
async 異步
sync 同步
3> 要執行異步的任務,就在全局隊列中執行便可
dispatch_async 異步執行控制不住前後順序
4> 關於GCD的隊列
全局隊列 dispatch_get_global_queue
參數:優先級 DISPATCH_QUEUE_PRIORITY_DEFAULT
始終是 0
1 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
可同步 可異步
串行隊列
dispatch_queue_t queue = dispatch_queue_create("myQueue", DISPATCH_QUEUE_SERIAL);
是建立獲得的,不能直接獲取
只能同步
主隊列 dispatch_get_main_queue
1 dispatch_async(dispatch_get_main_queue(), ^{ 2 NSLog(@"main - > %@", [NSThread currentThread]); 3 });
只能同歩
5> 異步和同步與方法名無關,與運行所在的隊列有關!
同步主要用來控制方法的被調用的順序