iOS多線程之NSoperation
目前在 iOS 和 OS X 中有兩套先進的同步 API 可供我們使用:NSOperation 和 GCD 。其中 GCD 是基于 C 的底層的 API ,而 NSOperation 則是 GCD 實現(xiàn)的 Objective-C API。 雖然 NSOperation 是基于 GCD 實現(xiàn)的, 但是并不意味著它是一個 GCD 的“dumbed-down” 版本, 相反,我們可以用NSOperation 輕易的實現(xiàn)一些 GCD 要寫大量代碼的事情。
NSOperation是一個抽象的基類,表示一個獨立的計算單元,可以為子類提供有用且線程安全的建立狀態(tài),優(yōu)先級,依賴和取消等操作。系統(tǒng)已經(jīng)給我們封裝了NSBlockOperation和NSInvocationOperation這兩個實體類。使用起來也非常簡單,不過我們更多的使用是自己繼承并定制自己的操作。
狀態(tài)
NSOperation提供了ready
cancelled
executing
finished
這幾個狀態(tài)變化,我們的開發(fā)也是必須處理自己關(guān)心的其中的狀態(tài)。這些狀態(tài)都是基于keypath的KVO通知決定,所以在你手動改變自己關(guān)心的狀態(tài)時,請別忘了手動發(fā)送通知。這里面每個屬性都是相互獨立的,同時只可能有一個狀態(tài)是YES。finished這個狀態(tài)在操作完成后請及時設(shè)置為YES,因為NSOperationQueue所管理的隊列中,只有isFinished為YES時才將其移除隊列,這點在內(nèi)存管理和避免死鎖很關(guān)鍵。
依賴
NSOperation中我們可以為操作分解為若干個小的任務(wù),通過添加他們之間的依賴關(guān)系進行操作,這點在設(shè)計上是很有意義的。比如我們最常用的圖片異步加載,第一步我們是去通過網(wǎng)絡(luò)進行加載,第二步我們可能需要對圖片進行下處理(調(diào)整大小或者壓縮保存)。我們可以直接調(diào)用- (void)addDependency:(NSOperation*)op
;
這點我們必須要注意的是不能添加相互依賴,像A依賴B,B依賴A,這樣會導(dǎo)致死鎖!還有一點必須要注意的時候,在每個操作完成時,請將isFinished設(shè)置為YES,不然后續(xù)的操作是不會開始執(zhí)行的。
執(zhí)行
執(zhí)行一個operation有兩種方法,第一種是自己手動的調(diào)用start這個方法,這種方法調(diào)用會在當(dāng)前調(diào)用的線程進行同步執(zhí)行,所以在主線程里面自己一定要小心的調(diào)用,不然就會把主線程給卡死。
第二種是將operation添加到operationQueue中去,這個也是我們用得最多的也是提倡的方法。NSOperationQueue會在我們添加進去operation的時候盡快進行執(zhí)行。當(dāng)然如果NSOperationQueue的maxConcurrentOperationCount如果設(shè)置為1的話,進相當(dāng)于FIFO了。
取消
NSOperation允許我們調(diào)用-(void)cancel取消一個操作的執(zhí)行。當(dāng)然,這個操作并不是我們所想象的取消。這個取消的步驟是這樣的,如果這個操作在隊列中沒有執(zhí)行,那么這個時候取消并將狀態(tài)finished設(shè)置為YES,那么這個時候的取消就是直接取消了。如果這個操作已經(jīng)在執(zhí)行了,那么我們只能等其操作完成。當(dāng)我們調(diào)用cancel方法的時候,他只是將isCancelled設(shè)置為YES。所以,在我們的操作中,我們應(yīng)該在每個操作開始前,或者在每個有意義的實際操作完成后,先檢查下這個屬性是不是已經(jīng)設(shè)置為YES。如果是YES,則后面操作都可以不用在執(zhí)行了。
completionBlock
iOS4后添加了這個block,在這個操作完成時,將會調(diào)用這個block一次,這樣也非常方便的讓我們對view進行更新或者添加自己的業(yè)務(wù)邏輯代碼。
優(yōu)先級
operationQueue有maxConcurrentOperationCount設(shè)置,當(dāng)隊列中operation很多時而你想讓后續(xù)的操作提前被執(zhí)行的時候,你可以為你的operation設(shè)置優(yōu)先級。
NSOperationQueuePriorityVeryLow = -8L,
NSOperationQueuePriorityLow = -4L,
NSOperationQueuePriorityNormal = 0,
NSOperationQueuePriorityHigh = 4,
NSOperationQueuePriorityVeryHigh = 8
使用
- 通過SDK自帶的NSBlockOperation和NSInvocationOperation。
- 自定義NSOperation的子類。
NSBlockOperation和NSInvocationOperation
其實除非必要,簡單的工作完全可以使用官方提供的NSOperation兩個NSInvocationOperation和NSBlockOperation來實現(xiàn)。
NSInvocationOperation:
NSInvocationOperation* theOp = [[NSInvocationOperation alloc]
initWithTarget:self
selector:@selector(myTaskMethod:)
object:data];
NSBlockOperation:
NSBlockOperation* theOp = [NSBlockOperation blockOperationWithBlock: ^{
NSLog(@"Beginning operation.\n");
// Do some work.
}];
自定義NSOperation 的子類
對于非并發(fā)的工作,只需要實現(xiàn)NSOperation子類的main方法。
-(void)main
{
@try
{
// 處理工作任務(wù)
}
@catch(...)
{
// 處理異常,但是不能再重新拋出異常
}
}
由于NSOperation的工作是可以取消Cancel的,那么你在main方法處理工作時就需要不斷輪詢[self isCancelled]
確認當(dāng)前的工作是否被取消了。
如果要支持并發(fā)工作,那么NSOperation子類需要至少override這四個方法:
- start
- isConcurrent
- isExecuting
- isFinished
注意你的實現(xiàn)要發(fā)出合適的KVO通知,因為如果你的NSOperation實現(xiàn)需要用到工作依賴從屬特性,而你的實現(xiàn)里沒有發(fā)出合適的“isFinished”KVO通知,依賴你的NSOperation就無法正常執(zhí)行。NSOperation支持KVO的屬性有:
? isCancelled
? isConcurrent
? isExecuting
? isFinished
? isReady
? dependencies
? queuePriority
? completionBlock