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ā)也是必須處理自己關心的其中的狀態(tài)聋呢。這些狀態(tài)都是基于keypath的KVO通知決定苗踪,所以在你手動改變自己關心的狀態(tài)時,請別忘了手動發(fā)送通知削锰。這里面每個屬性都是相互獨立的通铲,同時只可能有一個狀態(tài)是YES。finished這個狀態(tài)在操作完成后請及時設置為YES器贩,因為NSOperationQueue所管理的隊列中颅夺,只有isFinished為YES時才將其移除隊列,這點在內存管理和避免死鎖很關鍵蛹稍。
依賴
NSOperation中我們可以為操作分解為若干個小的任務吧黄,通過添加他們之間的依賴關系進行操作,這點在設計上是很有意義的稳摄。比如我們最常用的圖片異步加載稚字,第一步我們是去通過網(wǎng)絡進行加載,第二步我們可能需要對圖片進行下處理(調整大小或者壓縮保存)厦酬。我們可以直接調用- (void)addDependency:(NSOperation*)op
;
這點我們必須要注意的是不能添加相互依賴胆描,像A依賴B,B依賴A仗阅,這樣會導致死鎖昌讲!還有一點必須要注意的時候,在每個操作完成時减噪,請將isFinished設置為YES短绸,不然后續(xù)的操作是不會開始執(zhí)行的车吹。
執(zhí)行
執(zhí)行一個operation有兩種方法,第一種是自己手動的調用start這個方法醋闭,這種方法調用會在當前調用的線程進行同步執(zhí)行窄驹,所以在主線程里面自己一定要小心的調用,不然就會把主線程給卡死证逻。
第二種是將operation添加到operationQueue中去乐埠,這個也是我們用得最多的也是提倡的方法。NSOperationQueue會在我們添加進去operation的時候盡快進行執(zhí)行囚企。當然如果NSOperationQueue的maxConcurrentOperationCount如果設置為1的話丈咐,進相當于FIFO了。
取消
NSOperation允許我們調用-(void)cancel取消一個操作的執(zhí)行龙宏。當然棵逊,這個操作并不是我們所想象的取消。這個取消的步驟是這樣的银酗,如果這個操作在隊列中沒有執(zhí)行辆影,那么這個時候取消并將狀態(tài)finished設置為YES,那么這個時候的取消就是直接取消了花吟。如果這個操作已經(jīng)在執(zhí)行了秸歧,那么我們只能等其操作完成厨姚。當我們調用cancel方法的時候衅澈,他只是將isCancelled設置為YES。所以谬墙,在我們的操作中今布,我們應該在每個操作開始前,或者在每個有意義的實際操作完成后拭抬,先檢查下這個屬性是不是已經(jīng)設置為YES部默。如果是YES,則后面操作都可以不用在執(zhí)行了造虎。
completionBlock
iOS4后添加了這個block傅蹂,在這個操作完成時,將會調用這個block一次算凿,這樣也非常方便的讓我們對view進行更新或者添加自己的業(yè)務邏輯代碼份蝴。
優(yōu)先級
operationQueue有maxConcurrentOperationCount設置,當隊列中operation很多時而你想讓后續(xù)的操作提前被執(zhí)行的時候氓轰,你可以為你的operation設置優(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
{
// 處理工作任務
}
@catch(...)
{
// 處理異常怒医,但是不能再重新拋出異常
}
}
由于NSOperation的工作是可以取消Cancel的,那么你在main方法處理工作時就需要不斷輪詢[self isCancelled]
確認當前的工作是否被取消了奢讨。
如果要支持并發(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