NSOperation

NSOperation

  • 父類是NSObject

NSOperation與GCD隊列+操作的差別:

  • NSOperation將同步與異步函數封裝在內部;
  • 隊列由(串行隊列、主隊列、并發隊列、全局并發隊列)變為只分成主隊列和非主隊列:
    01. 同步與異步函數:
        1.1 同步函數:NSOperation中,任務的start方法是同步函數,start是讓任務在當前線程中同步串行執行,并且調用內部的main方法
            a  如果要讓任務在當前線程執行,就直接創建任務并start,而不是加入到隊列中

        1.2 異步函數:NSOperationQueue都是異步函數。加入到NSOperationQueue中的任務,內部會調用start方法,讓任務在當前線程中同步串行執行,并且調用內部的main方法

        1.3 NSBlockOperation追加任務也是在新開線程執行,是特例。

    02 NSOperation的隊列:
        1. 主隊列的任務在主線程中執行(同GCD);
        2. 非主隊列的任務同時具備串行和并發的能力,通過最大并發數設置隊列模式(串行?并發)。



1.NSOperation和GCD的對比?
1) GCD是純C語言的API,而操作隊列則是Object-C的對象。
2)在GCD中,任務用塊(block)來表示,而塊是個輕量級的數據結構;
   相反操作隊列中的『操作』NSOperation則是個更加重量級的Object-C對象。

操作隊列的好處有:
    1)NSOperationQueue可以方便的調用cancel方法來取消某個操作,而GCD中的任務是無法被取消的(安排好任務之后就不管了)。
    2)NSOperation可以方便的指定操作間的依賴關系。
    3)NSOperation可以通過KVO提供對NSOperation對象的精細控制。
    4)NSOperation可以方便的指定操作優先級。
    5)通過自定義NSOperation的子類可以實現操作重用。

NSOperation的使用

  • 任務執行完后不會銷毀,可繼續添加任務繼續執行;thread任務執行完就“死掉”,不能重新開始或者添加新的任務

(1)基本使用步驟

01 NSOperation本身是抽象類,只有它的子類才具備封裝操作的能力
02 三個子類分別是:NSBlockOperation、NSInvocationOperation以及自定義繼承自NSOperation的類
03 NSOperation和NSOperationQueue結合使用實現多線程并發具體步驟:
    3.1 先將需要執行的操作封裝到一個NSOperation對象中
    3.2 然后將NSOperation對象添加到NSOperationQueue中
    3.3 系統會自動將NSOperationQueue中的NSOperation取出來
    3.4 將取出的NSOperation封裝的操作放到新的線程中執行(開多少線程由系統管理)

(2)基本使用代碼

  01 NSInvocationOperation
    //1.封裝操作
    /*
     第一個參數:目標對象
     第二個參數:該操作要調用的方法,最多接受一個參數
     第三個參數:調用方法傳遞的參數,如果方法不接受參數,那么該值傳nil
     */
    NSInvocationOperation *operation = [[NSInvocationOperation alloc]
                                        initWithTarget:self selector:@selector(run) object:nil];

    //2.啟動操作
    [operation start];//開始調用run方法

-------------------------------------------------
  02 NSBlockOperation
    //1.封裝操作
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{

        //在主線程中執行
        NSLog(@"---download1--%@",[NSThread currentThread]);
    }];

    //2.追加操作,追加的操作在子線程中執行
    [operation addExecutionBlock:^{
        NSLog(@"---download2--%@",[NSThread currentThread]);
    }];

    //3.啟動執行操作
    [operation start];

----------------------------------------------
 03 自定義NSOperation
    //如何封裝操作?
    //自定義的NSOperation,通過重寫內部的main方法實現封裝操作
    //重寫- (void)main方法的注意點
自己創建自動釋放池(因為如果是異步操作,無法訪問主線程的自動釋放池)
    //經常通過- (BOOL)isCancelled方法檢測操作是否被取消,對取消做出響應

//優點:
//1.代碼簡潔
//2.提高代碼的復用性
//3.信息隱蔽
自定義FZQOperation.m中
    -(void)main
    {
        NSLog(@"--main--%@",[NSThread currentThread]);
    }

    //如何使用?
    //1.實例化一個自定義操作對象
    XMGOperation *op = [[XMGOperation alloc]init];

    //2.執行操作
    [op start];
    //start方法內部會調用main方法,所以可以把線程調用的代碼放在內部的main方法中

NSOperationQueue基本使用

(1)NSOperation中的兩種隊列

01 主隊列 通過mainQueue獲得,凡是放到主隊列中的任務都將在主線程執行
02 非主隊列 直接alloc init出來的隊列。非主隊列同時具備了并發和串行的功能,通過設置最大并發數屬性來控制任務是并發執行還是串行執行。具備新開線程能力

(2)相關代碼

    01 自定義NSOperation
-(void)customOperation
{
    //1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封裝操作
    //好處:1.信息隱蔽
    //2.代碼復用

    XMGOperation *op1 = [[XMGOperation alloc]init];
    XMGOperation *op2 = [[XMGOperation alloc]init];

    //3.添加操作到隊列中
    [queue addOperation:op1];
    [queue addOperation:op2];
}

    02 NSBlockOperation
- (void)block
{
    //1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封裝操作
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"1----%@",[NSThread currentThread]);
    }];

    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"2----%@",[NSThread currentThread]);

    }];

    [op2 addExecutionBlock:^{
        NSLog(@"3----%@",[NSThread currentThread]);
    }];

    [op2 addExecutionBlock:^{
        NSLog(@"4----%@",[NSThread currentThread]);
    }];

    //3.添加操作到隊列中
    [queue addOperation:op1];
    [queue addOperation:op2];

    //補充:簡便方法
    [queue addOperationWithBlock:^{
        NSLog(@"5----%@",[NSThread currentThread]);
    }];

}

    03 NSInvocationOperation
- (void)invocation
{
    /*
     GCD中的隊列:
     串行隊列:自己創建的,主隊列
     并發隊列:自己創建的,全局并發隊列

     NSOperationQueue
     主隊列:[NSOperationQueue mainqueue];凡是放在主隊列中的操作都在主線程中串行執行
     非主隊列:[[NSOperationQueue alloc]init],并發和串行,默認是并發執行的
     */

    //1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封裝操作
    NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download1) object:nil];

    NSInvocationOperation *op2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download2) object:nil];

    //3.把封裝好的操作添加到隊列中
    [queue addOperation:op1];
    [queue addOperation:op2];
}

NSOperation其它用法

(1)設置最大并發數【控制任務并發和串行】

//1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.設置最大并發數
    //注意點:該屬性需要在任務添加到隊列中之前進行設置
    //該屬性控制隊列是串行執行還是并發執行
    //如果最大并發數等于1,那么該隊列是串行的,如果大于1那么是并行的
    //系統的最大并發數有個默認的值,為-1,如果該屬性設置為0,那么不會執行任何任務
    queue.maxConcurrentOperationCount = 2;

(2)暫停和恢復以及取消

    //設置暫停和恢復
    //suspended設置為YES表示暫停,suspended設置為NO表示恢復
    //暫停表示不繼續執行隊列中的下一個任務,暫停操作是可以恢復的
    self.queue.suspended = !self.queue.suspended;


    //取消隊列里面的所有操作
    //取消之后,當前正在執行的操作的下一個操作將不再執行,而且永遠都不在執行,就像后面的所有任務都從隊列里面移除了一樣
    //取消操作是不可以恢復的
    [self.queue cancelAllOperations];

---------自定義NSOperation取消操作-----------------------

-(void)main
{
    //耗時操作1
    for (int i = 0; i<1000; i++) {
        NSLog(@"任務1-%d--%@",i,[NSThread currentThread]);
    }
    NSLog(@"+++++++++++++++++++++++++++++++++");

    //蘋果官方建議,每當執行完一次耗時操作之后,就查看一下當前隊列是否為取消狀態,如果是,那么就直接退出
    //好處是可以提高程序的性能
    if (self.isCancelled) {
        return;
    }

    //耗時操作2
    for (int i = 0; i<1000; i++) {
        NSLog(@"任務1-%d--%@",i,[NSThread currentThread]);
    }

    NSLog(@"+++++++++++++++++++++++++++++++++");
}

(3)操作依賴

01 NSOperation之間可以設置依賴來保證執行順序
- (void)addOperation:(NSOperation *)op;
- (void)addOperationWithBlock:(void (^)(void))block;

//添加操作依賴,B在A之后執行
[operationB addDependency:operationA];

02 可以在不同queue的NSOperation之間創建依賴關系

03注意:不能相互依賴
比如A依賴B,B依賴A,否則AB及依賴AB的操作都不會執行(不會崩潰)
    /*
    [download addDependency:op4];
    [op4 addDependency:download];
    */

(4)操作的監聽

可以監聽一個操作的執行完畢
- (void (^)(void))completionBlock;
- (void)setCompletionBlock:(void (^)(void))block;
  • 2.4 NSOperation實現線程間通信

(1)開子線程下載圖片

 //1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.使用簡便方法封裝操作并添加到隊列中
    [queue addOperationWithBlock:^{

        //3.在該block中下載圖片
        NSURL *url = [NSURL URLWithString:@"http://news.51sheyuan.com/uploads/allimg/111001/133442IB-2.jpg"];
        NSData *data = [NSData dataWithContentsOfURL:url];
        UIImage *image = [UIImage imageWithData:data];
        NSLog(@"下載圖片操作--%@",[NSThread currentThread]);

        //4.回到主線程刷新UI
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            self.imageView.image = image;
            NSLog(@"刷新UI操作---%@",[NSThread currentThread]);
        }];
    }];

(2)下載多張圖片合成綜合案例(設置操作依賴)

//02 綜合案例
- (void)download2
{
    NSLog(@"----");
    //1.創建隊列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封裝操作下載圖片1
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{

        NSURL *url = [NSURL URLWithString:@"http://h.hiphotos.baidu.com/zhidao/pic/item/6a63f6246b600c3320b14bb3184c510fd8f9a185.jpg"];
        NSData *data = [NSData dataWithContentsOfURL:url];

        //拿到圖片數據
        self.image1 = [UIImage imageWithData:data];
    }];


    //3.封裝操作下載圖片2
    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSURL *url = [NSURL URLWithString:@"http://pic.58pic.com/58pic/13/87/82/27Q58PICYje_1024.jpg"];
        NSData *data = [NSData dataWithContentsOfURL:url];

        //拿到圖片數據
        self.image2 = [UIImage imageWithData:data];
    }];

    //4.合成圖片
    NSBlockOperation *combine = [NSBlockOperation blockOperationWithBlock:^{

        //4.1 開啟圖形上下文
        UIGraphicsBeginImageContext(CGSizeMake(200, 200));

        //4.2 畫image1
        [self.image1 drawInRect:CGRectMake(0, 0, 200, 100)];

        //4.3 畫image2
        [self.image2 drawInRect:CGRectMake(0, 100, 200, 100)];

        //4.4 根據圖形上下文拿到圖片數據
        UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
//        NSLog(@"%@",image);

        //4.5 關閉圖形上下文
        UIGraphicsEndImageContext();

        //7.回到主線程刷新UI
        [[NSOperationQueue mainQueue]addOperationWithBlock:^{
            self.imageView.image = image;
            NSLog(@"刷新UI---%@",[NSThread currentThread]);
        }];

    }];

    //5.設置操作依賴
    [combine addDependency:op1];
    [combine addDependency:op2];

    //6.添加操作到隊列中執行
    [queue addOperation:op1];
    [queue addOperation:op2];
    [queue addOperation:combine];
    }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容