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];