iOS開發-多線程NSOperation和NSOperationQueue
NSThread能直觀地控制線程對象,不過需要自己管理線程的生命週期,線程同步,用起來比較繁瑣,而且比較容易出錯。不過Apple給出了自己的解決方案NSOperation,它本身是抽象基類,因此必須使用它的子類,使用NSOperation子類的方式有NSInvocationOperation和NSBlockOperation兩種方式,先補充一下NSThread的用法: NSThread擷取當前線程: [NSThread currentThread] performSelectorInBackground可以更新UI,不建議使用: - (IBAction)update:(id)sender { [self performSelectorInBackground:@selector(changeImage) withObject:nil]; }圖片背景更新: -(void)changeImage{ NSLog(@"線程執行完之後更新圖片"); self.myImageView.image=[UIImage imageNamed:[NSString stringWithFormat:@"Thread2.jpg"]];}NSInvocationOperation和NSBlockOperation 這兩種方式都很簡單,其中NSInvocation的調用方式類似於NSThread,NSBlockOperation如果對Block有一點瞭解就可以,如果不明白的可以參考本人之前的Block文章 Object-C-代碼塊Block回顧,那麼接下來的使用方式就很簡單: 先來看下NSInvocationOperation的執行個體化方式: //初始化NSInvocationOperation *myInvocationOperation= [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(operationTaskMethod) object:nil];//啟動[myInvocationOperation start];調用方法: -(void)operationTaskMethod{ NSLog(@"NSInvocationOperation初始化執行"); }NSBlockOperation的方式: NSBlockOperation *blockOperation=[NSBlockOperation blockOperationWithBlock:^{ NSLog(@"BlockOperation塊執行"); }]; [blockOperation start];兩種方式很方便,這個時候可以使用NSOperationQueue作為一個隊列將線程包含在一起,首先定義一個NSOperationQuene: @property (strong,nonatomic) NSOperationQueue *myOperationQuene; 這個時候需要調用: NSInvocationOperation *myInvocationOperation= [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(operationTaskMethod) object:nil]; NSBlockOperation *blockOperation=[NSBlockOperation blockOperationWithBlock:^{ NSLog(@"BlockOperation塊執行");}]; self.myOperationQuene=[[NSOperationQueue alloc]init]; [self.myOperationQuene addOperation:myInvocationOperation]; [self.myOperationQuene addOperation:blockOperation]; 上面最後的結果不確定,線程執行的順序沒法確定,如果想確定的按照順序執行,需要添加一個依賴: [blockOperation addDependency:myInvocationOperation]; 添加依賴之後的,每次輸出的結果一定是這樣的: 2015-02-11 07:56:13.457 ThreadDemo[657:15033] NSInvocationOperation初始化執行2015-02-11 07:56:13.457 ThreadDemo[657:15034] BlockOperation塊執行 自訂NSOperation 每次一看到自訂,就感覺瞬間有了檔次,然後參考一下前人的經驗,不過網上的部落格有的不好說,那感覺就像我像我只想吃一個雞腿,確拿到了一個雞腿堡,需要的不需要的都要自己一起吸收。NSInvocationOperation和NSBlockOperation這兩種方式不能滿足業務需求,這個時候需要自訂的NSOperation,自訂的有兩種分為非並發(NonConcurrent)和並發(Concurrent)兩種形式,本文介紹非並發形式。 建立一個繼承自NSOperation的MyCustomOperation,然後實現一下main方法: //// MyCustomOperation.h// ThreadDemo//// Created by keso on 15/2/11.// Copyright (c) 2015年 keso. All rights reserved.// #import <Foundation/Foundation.h> @interface MyCustomOperation : NSOperation @property (strong,nonatomic) NSString *customdata; -(void)initData:(NSString *)data; @end NSOperation對象需要定期地調用isCancelled方法檢測操作是否已經被取消,如果返回YES(表示已取消),則立即退出執行回收記憶體資源。所有NSOperation子類,一般用於代碼比較容易終止的地方, 在迴圈的每次迭代過程中,如果每個迭代相對較長可能需要調用多次和沒有執行工作之前調用。 //// MyCustomOperation.m// ThreadDemo//// Created by keso on 15/2/10.// Copyright (c) 2015年 keso. All rights reserved.// #import "MyCustomOperation.h" @implementation MyCustomOperation - (void)initData:(NSString *)data{ if (self ==[super init]) _customdata= data;}- (void)main { @try { BOOL isDone = NO; NSLog(@"迴圈之前的調用"); while (![self isCancelled] && !isDone) { // Do some work and set isDone to YES when finished NSLog(@"已經運行成功了"); isDone=YES; } } @catch(...) { NSLog(@"出現異常,請檢查代碼~"); }} @end 如果需要調用定義的NSOPeration執行個體化之後Start即可: MyCustomOperation *customOperation=[[MyCustomOperation alloc] init]; [customOperation start];