多線程之NSOperation和NSOperationQueue

来源:http://www.cnblogs.com/zzuliliu/archive/2016/05/06/5467160.html
-Advertisement-
Play Games

這篇文章里我將不過多的談及理論知識,這些東西會的自然會,不會的,看多了也是雲里霧裡。下麵我講更多的用代碼+註釋的方式來講如何使用NSOperation和NSOperationQueue。 1、NSOperation。是抽象類,不能夠直接使用,而是使用子類NSInvocationOperation和N ...


  這篇文章里我將不過多的談及理論知識,這些東西會的自然會,不會的,看多了也是雲里霧裡。下麵我講更多的用代碼+註釋的方式來講如何使用NSOperation和NSOperationQueue。

  1、NSOperation。是抽象類,不能夠直接使用,而是使用子類NSInvocationOperation和NSBlockOperation來實際執行任務。NSOperation本身和多線程是沒有任何關係的,她只是封裝了一個代碼段和數據去實現一個功能。

  1.1、NSInvocationOperation,基於一個對象和selector來創建操作。看下麵的代碼:

 1 - (void)invocationOperation {
 2     
 3     NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(startOpration) object:nil];
 4     [operation setCompletionBlock:^{
 5         NSLog(@"執行完成,是否主線程:%@", [NSThread isMainThread] == 1 ? @"YES" : @"NO");
 6     }];
 7     [operation start];  //啟動任務
 8     [operation cancel]; //取消任務
 9     [operation isExecuting];    //任務是否在執行
10     [operation isFinished];     //任務是否已經結束
11 }
12 
13 - (void)startOpration {
14     NSLog(@"開始執行,是否主線程:%@", [NSThread isMainThread] == 1 ? @"YES" : @"NO");
15 }

  首先是創建了一個operation,並給operation添加了一個需要執行的方法:startOpration。其中Line4中的方法將會在operation執行完成後執行,當然不是必須的,如果需要在operation執行完進行一些操作,可以寫上這個方法。

  下麵是執行的結果:

2016-05-06 20:28:19.577 NSOperation[1725:364426] 開始執行,是否主線程:YES
2016-05-06 20:28:19.577 NSOperation[1725:364462] 執行完成,是否主線程:NO

  根據上面執行的結果,我們可以發現, startOpration 這個方法是在主線程執行的。至於Line4~Line6是在子線程執行的,則不在今天的討論內容中,略過。

  1.2、NSBlockOperation。相對於NSInvocationOperation,NSBlockOperation則是將selector中需要調用的方法使用Block進行了封裝,使用起來更加的方便。關於Block的使用,不明白的同學可以參考我的上一篇博客。

  NSBlockOperation對象能夠併發的執行一個或多個Block對象,所有相關的Block都執行完成之後,操作才算完成。下麵看代碼:

 1 - (void)blockOperation {
 2     //創建一個操作
 3     NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
 4         NSLog(@"%d,%@,是否主線程:%@", __LINE__, [NSThread currentThread], [NSThread isMainThread] == 1 ? @"YES" : @"NO");
 5     }];
 6     //通過addExecutionBlock方法添加Block操作
 7     [blockOperation addExecutionBlock:^{
 8         NSLog(@"%d,%@,是否主線程:%@", __LINE__, [NSThread currentThread], [NSThread isMainThread] == 1 ? @"YES" : @"NO");
 9     }];
10     [blockOperation addExecutionBlock:^{
11         NSLog(@"%d,%@,是否主線程:%@", __LINE__, [NSThread currentThread], [NSThread isMainThread] == 1 ? @"YES" : @"NO");
12     }];
13     [blockOperation addExecutionBlock:^{
14         NSLog(@"%d,%@,是否主線程:%@", __LINE__, [NSThread currentThread], [NSThread isMainThread] == 1 ? @"YES" : @"NO");
15     }];
16     [blockOperation addExecutionBlock:^{
17         NSLog(@"%d,%@,是否主線程:%@", __LINE__, [NSThread currentThread], [NSThread isMainThread] == 1 ? @"YES" : @"NO");
18     }];
19     //執行操作
20     [blockOperation start];
21     //上面的幾個Block是併發執行的,將會在不同的線程中執行,
22 }

  下麵是執行的結果:

2016-05-06 20:58:28.585 NSOperation[1754:389977] 48,<NSThread: 0x7f97cad07c30>{number = 1, name = main},是否主線程:YES
2016-05-06 20:58:28.585 NSOperation[1754:390024] 44,<NSThread: 0x7f97cae223c0>{number = 2, name = (null)},是否主線程:NO
2016-05-06 20:58:28.585 NSOperation[1754:390042] 54,<NSThread: 0x7f97cac0eb60>{number = 4, name = (null)},是否主線程:NO
2016-05-06 20:58:28.586 NSOperation[1754:389977] 57,<NSThread: 0x7f97cad07c30>{number = 1, name = main},是否主線程:YES
2016-05-06 20:58:28.585 NSOperation[1754:390030] 51,<NSThread: 0x7f97cad06530>{number = 3, name = (null)},是否主線程:NO

  可以從上面的執行結果中,很容易的發現,程式是並行的執行。而創建的操作,也就是:Line4是在主線程中執行,而其他通過通過 addExecutionBlock 方法添加的Block操作則是在不同的線程中執行。

  通過上面的說明,我要總結的是:NSOperation以及子類,只是一個操作,本身無主線程和子線程的區分,可以在任意線程中使用。也就是,在主線程中創建的操作將在主線程中執行,在子線程中創建的操作將在子線程中執行。而實現多線程可以配合NSOperationQueue來實現。下麵來介紹NSOperationQueue。

  2.NSOperationQueue,是操作隊列,它用來管理一組NSOperation對象的執行,會根據需要自動為NSOperation對象開闢適合數量的線程,以完成任務的並行執行。下麵上代碼:

 1 - (void)operationQueue {
 2     //創建一個任務隊列,alloc(new)出來的任務隊列將會在子線程中執行,並且是併發執行
 3     NSOperationQueue *queue = [[NSOperationQueue alloc] init];
 4     //創建操作NSInvocationOperation
 5     for (int i = 0; i < 5; i++) {
 6         NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(startOpration) object:nil];
 7         [queue addOperation:operation];
 8     }
 9     //創建NSBlockOperation操作
10     for (int i = 0; i < 5; i++) {
11         NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
12             NSLog(@"%d,%@,%d", __LINE__, [NSThread currentThread], [NSThread isMainThread]);
13         }];
14         //將操作添加到隊列中
15         [queue addOperation:blockOperation];
16     }
17     //設置最大併發數,當把最大併發數設置1時,此時隊列相當於串列執行。
18     queue.maxConcurrentOperationCount = 3;
19     //添加依賴關係
20     NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
21         NSLog(@"這是任務1,依賴於任務2");
22     }];
23     NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
24         NSLog(@"這是任務2");
25     }];
26     //添加依賴關係
27     //添加依賴關係時,註意不能添加互相依賴,如A依賴B,B依賴A
28     [op1 addDependency:op2];
29     //添加依賴關係必須在任務添加到隊列之前設置
30     [queue addOperation:op1];
31     [queue addOperation:op2];
32     //回到主線程的操作
33     [[NSOperationQueue mainQueue] addOperationWithBlock:^{
34         NSLog(@"%d,%@,%d", __LINE__, [NSThread currentThread], [NSThread isMainThread]);
35     }];
36 }

  下麵是執行的結果:

 1 2016-05-06 21:15:11.625 NSOperation[1770:402666] 開始執行,是否主線程:NO
 2 2016-05-06 21:15:11.625 NSOperation[1770:402633] 開始執行,是否主線程:NO
 3 2016-05-06 21:15:11.625 NSOperation[1770:402642] 開始執行,是否主線程:NO
 4 2016-05-06 21:15:11.625 NSOperation[1770:402654] 開始執行,是否主線程:NO
 5 2016-05-06 21:15:11.625 NSOperation[1770:402664] 開始執行,是否主線程:NO
 6 2016-05-06 21:15:11.626 NSOperation[1770:402665] 75,<NSThread: 0x7f852bc099f0>{number = 2, name = (null)},0
 7 2016-05-06 21:15:11.627 NSOperation[1770:402667] 75,<NSThread: 0x7f852bf0db10>{number = 3, name = (null)},0
 8 2016-05-06 21:15:11.628 NSOperation[1770:402633] 75,<NSThread: 0x7f852bc46ee0>{number = 4, name = (null)},0
 9 2016-05-06 21:15:11.628 NSOperation[1770:402666] 75,<NSThread: 0x7f852bc4fe00>{number = 6, name = (null)},0
10 2016-05-06 21:15:11.628 NSOperation[1770:402642] 75,<NSThread: 0x7f852be0a130>{number = 5, name = (null)},0
11 2016-05-06 21:15:11.629 NSOperation[1770:402633] 這是任務2
12 2016-05-06 21:15:11.629 NSOperation[1770:402633] 這是任務1,依賴於任務2
13 2016-05-06 21:15:11.631 NSOperation[1770:402393] 97,<NSThread: 0x7f852bd05190>{number = 1, name = main},1

  從上面的執行結果中,我們可以得到幾個結論:1.首先隊列中的任務都是在子線程中執行的;2.隊列中的任務是併發執行的。而常用到的添加依賴關係,以及操作完成之後回到主線程的方法在上面的代碼中都有體現。

  需要說明的是,一旦將operation添加到隊列中後,隊列就擁有了這個操作,操作將不能從隊列中刪除,只能取消一個操作,或者取消所有的操作。取消一個操作的方法:

[operation cancel];

  取消所有操作的方法:

[queue cancelAllOperations];

  如果想要暫停操作的執行可以使用方法: [queue setSuspended:YES]; 來實現。但是暫停不會導致正在執行的操作被暫停,只會阻止隊列調度新的操作執行。可以通過方法: [queue setSuspended:NO]; 來繼續隊列的執行。

  3、最後做一個簡單的總結。NSInvocationOperation 和 NSBlockOperation的功能是開啟一個線程任務,這個線程任務處於子線程和主線程取決於線程任務是在哪個線程中開闢的。這些任務可以使用線程任務隊列NSOperationQueue來管理,當new一個隊列時,隊列所管理的任務都在子線程中執行.當使用[NSOperationQueue mainQueue]時,隊列管理的任務在主線程中執行。

  以上就是這篇文章的多有內容,如果有不對或者不足的地方,請大家在評論中指出,大家一起討論。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這次我們來講解第三節知識,考慮了下,先不去講什麼理論了,畢竟網上一搜一大堆,而且理論真心看不太懂啊!!! 今天我們就直接上實例嘍! 大家HIGH起來!!!(想了好久,還是沒捨得刪這句話) 1.根據下圖配置自己的文件 2.我們先用原始方法 額,忘記告訴大家做什麼項目了,那就在這補上吧,咱們先做一個小D ...
  • 在一個類的實例記憶體被釋放之前,析構方法被立即調用。使用deinit關鍵字來聲明析構方法,類似於構造方法用init來聲明。析構方法只適用於類類型。 析構方法原理 Swift會自動釋放不再需要的實例以釋放資源。如自動引用計數那一章描述,Swift通過自動引用計數(ARC)處理實例的記憶體管理。不需要手動的 ...
  • 在Android中,視圖控制項大致被分為兩類,即ViewGroup和View,ViewGroup控制項作為父控制項,包含並管理著子View,通過ViewGroup和View便形成了控制項樹,各個ViewGoup對象和View對象就是控制項樹中的節點。在控制項樹中,以樹的深度來遍歷查找對應的控制項元素,同時,上層控 ...
  • 源代碼管理工具SVN是一款非常強大的源代碼管理工具,現在國內70%-90%的公司都在使用SVN來管理源代碼,下麵就讓小編給大家著重介紹一下SVN的使用,SVN的使用主要分為下麵幾塊。 SVN的使用環境 伺服器端的配置 客戶端軟體的使用 客戶端圖形界面工具的使用 SVN的目錄規範及使用實例 1.SVN ...
  • Android做分享功能百度一下就兩種方案,其一是用系統原生的Activity,最終彈出一個對話框,下麵這種的還好,像右圖的那種就嫌棄了,上面提供的應用也相對雜,還記得有次測試還給鄙人提了個Bug:建議分享中多列舉常用的App如QQ微信等。 那另外一種方案就彌補了上面兩種不足,使用第三方的,如Sha ...
  • 程式 :由源代碼生成的可執行應用 QQ.app 進程 :一個正在運行的程式可以看作是一個進程,擁有獨立運行所需的全部資源 線程 :程式中獨立運行的代碼段 一個進程是由一個或多個線程組成。進程只負責資源的調度和分配,線程才是程式真正的執行單元,負責代碼的執行。 單線程 :每個正在運行的程式就是上面說的 ...
  • 一 數字 1.使用Foundation.h可以直接導入所有的頭文件。 在XCode中,想查看某個方法幫助,可以將游標放在方法上,按住option鍵同時單擊即可。 官方文檔:https://developer.apple.com/library/mac/navigation/ 2.數字對象 objec ...
  • 在開發中,我們經常需要更新列表,並將列表拉倒最底部,比如發表微博,聊天界面等等, 這裡有兩種辦法,第一種,使用scrollTo(): 第一種實現相對比較麻煩,更推薦使用第二種方式,使用fullScrol() 下麵我們看一下這個函數: scrollView.fullScroll(ScrollView. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...