你好,歡迎來到IOS教程網

 Ios教程網 >> IOS編程開發 >> IOS開發綜合 >> iOS開發-多線程NSOperation和NSOperationQueue

iOS開發-多線程NSOperation和NSOperationQueue

編輯:IOS開發綜合
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];
  1. 上一頁:
  2. 下一頁:
蘋果刷機越獄教程| IOS教程問題解答| IOS技巧綜合| IOS7技巧| IOS8教程
Copyright © Ios教程網 All Rights Reserved