標籤:
上一篇文章稍微提及了一下NSThread的使用,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];
參考資料:https://developer.apple.com/library/ios/documentation/General/Conceptual/ConcurrencyProgrammingGuide/OperationObjects/OperationObjects.html#//apple_ref/doc/uid/TP40008091-CH101-SW6
iOS開發-多線程NSOperation和NSOperationQueue