欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

iOS多线程应用开发中使用NSOperation类的基本方法

程序员文章站 2023-02-07 12:08:32
一、nsoperation简介 1.简单说明 nsoperation的作⽤:配合使用nsoperation和nsoperationqueue也能实现多线程...

一、nsoperation简介

1.简单说明

nsoperation的作⽤:配合使用nsoperation和nsoperationqueue也能实现多线程编程

nsoperation和nsoperationqueue实现多线程的具体步骤:

(1)先将需要执行的操作封装到一个nsoperation对象中

(2)然后将nsoperation对象添加到nsoperationqueue中

(3)系统会⾃动将nsoperationqueue中的nsoperation取出来

(4)将取出的nsoperation封装的操作放到⼀条新线程中执⾏

 2.nsoperation的子类

nsoperation是个抽象类,并不具备封装操作的能力,必须使⽤它的子类

使用nsoperation⼦类的方式有3种:

(1)nsinvocationoperation

(2)nsblockoperation

(3)自定义子类继承nsoperation,实现内部相应的⽅法

二、 具体说明

1.nsinvocationoperation子类

创建对象和执行操作:

复制代码 代码如下:

//创建操作对象,封装要执行的任务
    //nsinvocationoperation   封装操作
    nsinvocationoperation *operation=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test) object:nil];
   
    //执行操作
    [operation start];

说明:一旦执⾏操作,就会调用target的test方法

代码示例:

复制代码 代码如下:

//
//  yyviewcontroller.m
//  01-nsoperation基本1
//
//  created by 孔医己 on 14-6-25.
//  copyright (c) 2014年 itcast. all rights reserved.
//

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];
   
    //nsoperation:抽象类,不具备封装功能
   
    //创建操作对象,封装要执行的任务
    //nsinvocationoperation   封装操作
    nsinvocationoperation *operation=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test) object:nil];
   
    //执行操作
    [operation start];

}

-(void)test
{
   
    nslog(@"--test--%@--",[nsthread currentthread]);
}
@end


打印查看:

iOS多线程应用开发中使用NSOperation类的基本方法

注意:操作对象默认在主线程中执行,只有添加到队列中才会开启新的线程。即默认情况下,如果操作没有放到队列中queue中,都是同步执行。只有将nsoperation放到一个nsoperationqueue中,才会异步执行操作

2.nsblockoperation子类

创建对象和添加操作:

复制代码 代码如下:

//创建nsblockoperation操作对象
    nsblockoperation *operation=[nsblockoperation blockoperationwithblock:^{
        //......
    }];
   
    //添加操作
    [operation addexecutionblock:^{
        //....
    }];

代码示例:

代码1:

复制代码 代码如下:

//
//  yyviewcontroller.m
//  02-nstherad基本2
//
//  created by 孔医己 on 14-6-25.
//  copyright (c) 2014年 itcast. all rights reserved.
//

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];
   
    //创建nsblockoperation操作对象
    nsblockoperation *operation=[nsblockoperation blockoperationwithblock:^{
        nslog(@"nsblockoperation------%@",[nsthread currentthread]);
    }];
   
   
    //开启执行操作
    [operation start];
}
@end


打印查看:

iOS多线程应用开发中使用NSOperation类的基本方法

代码2:

复制代码 代码如下:

//
//  yyviewcontroller.m
//  02-nstherad基本2
//
//  created by 孔医己 on 14-6-25.
//  copyright (c) 2014年 itcast. all rights reserved.
//

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];
   
    //创建nsblockoperation操作对象
    nsblockoperation *operation=[nsblockoperation blockoperationwithblock:^{
        nslog(@"nsblockoperation------%@",[nsthread currentthread]);
    }];
   
    //添加操作
    [operation addexecutionblock:^{
        nslog(@"nsblockoperation1------%@",[nsthread currentthread]);
    }];
   
    [operation addexecutionblock:^{
        nslog(@"nsblockoperation2------%@",[nsthread currentthread]);
    }];
   
    //开启执行操作
    [operation start];
}
@end


iOS多线程应用开发中使用NSOperation类的基本方法

注意:只要nsblockoperation封装的操作数 > 1,就会异步执行操作

3.nsoperationqueue

nsoperationqueue的作⽤:nsoperation可以调⽤start⽅法来执⾏任务,但默认是同步执行的

如果将nsoperation添加到nsoperationqueue(操作队列)中,系统会自动异步执行nsoperation中的操作

添加操作到nsoperationqueue中,自动执行操作,自动开启线程

复制代码 代码如下:

//创建nsoperationqueue
    nsoperationqueue * queue=[[nsoperationqueue alloc]init];
    //把操作添加到队列中
    //第一种方式
    [queue addoperation:operation1];
    [queue addoperation:operation2];
    [queue addoperation:operation3];
    //第二种方式
    [queue addoperationwithblock:^{
        nslog(@"nsblockoperation3--4----%@",[nsthread currentthread]);
    }];

复制代码 代码如下:

- (void)addoperation:(nsoperation *)op;
- (void)addoperationwithblock:(void (^)(void))block;

代码示例:
复制代码 代码如下:

//
//  yyviewcontroller.m
//  03-nsoperation基本3
//
//  created by 孔医己 on 14-6-25.
//  copyright (c) 2014年 itcast. all rights reserved.
//

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];

    //创建nsinvocationoperation对象,封装操作
    nsinvocationoperation *operation1=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test1) object:nil];
    nsinvocationoperation *operation2=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test2) object:nil];
    //创建对象,封装操作
    nsblockoperation *operation3=[nsblockoperation blockoperationwithblock:^{
        nslog(@"nsblockoperation3--1----%@",[nsthread currentthread]);
    }];
    [operation3 addexecutionblock:^{
        nslog(@"nsblockoperation3--2----%@",[nsthread currentthread]);
    }];
   
    //创建nsoperationqueue
    nsoperationqueue * queue=[[nsoperationqueue alloc]init];
    //把操作添加到队列中
    [queue addoperation:operation1];
    [queue addoperation:operation2];
    [queue addoperation:operation3];
}


复制代码 代码如下:

-(void)test1
{
    nslog(@"nsinvocationoperation--test1--%@",[nsthread currentthread]);
}

-(void)test2
{
    nslog(@"nsinvocationoperation--test2--%@",[nsthread currentthread]);
}

@end


打印效果:

iOS多线程应用开发中使用NSOperation类的基本方法

注意:系统自动将nsoperationqueue中的nsoperation对象取出,将其封装的操作放到一条新的线程中执行。上面的代码示例中,一共有四个任务,operation1和operation2分别有一个任务,operation3有两个任务。一共四个任务,开启了四条线程。通过任务执行的时间全部都是273可以看出,这些任务是并行执行的。

提示:队列的取出是有顺序的,与打印结果并不矛盾。这就好比,选手a,bc虽然起跑的顺序是先a,后b,然后c,但是到达终点的顺序却不一定是a,b在前,c在后。
下面使用for循环打印,可以更明显的看出任务是并发执行的。

代码示例:

复制代码 代码如下:

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];

    //创建nsinvocationoperation对象,封装操作
    nsinvocationoperation *operation1=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test1) object:nil];
    nsinvocationoperation *operation2=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test2) object:nil];
    //创建对象,封装操作
    nsblockoperation *operation3=[nsblockoperation blockoperationwithblock:^{
        for (int i=0; i<5; i++) {
            nslog(@"nsblockoperation3--1----%@",[nsthread currentthread]);
        }
    }];
    [operation3 addexecutionblock:^{
        for (int i=0; i<5; i++) {
        nslog(@"nsblockoperation3--2----%@",[nsthread currentthread]);
        }
    }];
   
    //创建nsoperationqueue
    nsoperationqueue * queue=[[nsoperationqueue alloc]init];
    //把操作添加到队列中
    [queue addoperation:operation1];
    [queue addoperation:operation2];
    [queue addoperation:operation3];
}

-(void)test1
{
    for (int i=0; i<5; i++) {
    nslog(@"nsinvocationoperation--test1--%@",[nsthread currentthread]);
    }
}

-(void)test2
{
    for (int i=0; i<5; i++) {
    nslog(@"nsinvocationoperation--test2--%@",[nsthread currentthread]);
    }
}

@end


iOS多线程应用开发中使用NSOperation类的基本方法

三、并发数
(1)并发数:同时执⾏行的任务数.比如,同时开3个线程执行3个任务,并发数就是3
(2)最大并发数:同一时间最多只能执行的任务的个数。
(3)最⼤大并发数的相关⽅方法

复制代码 代码如下:

- (nsinteger)maxconcurrentoperationcount;
- (void)setmaxconcurrentoperationcount:(nsinteger)cnt;


说明:如果没有设置最大并发数,那么并发的个数是由系统内存和cpu决定的,可能内存多久开多一点,内存少就开少一点。
注意:num的值并不代表线程的个数,仅仅代表线程的id。
提示:最大并发数不要乱写(5以内),不要开太多,一般以2~3为宜,因为虽然任务是在子线程进行处理的,但是cpu处理这些过多的子线程可能会影响ui,让ui变卡。

四、队列的取消,暂停和恢复
 (1)取消队列的所有操作

复制代码 代码如下:

 - (void)cancelalloperations;

提⽰:也可以调用nsoperation的- (void)cancel⽅法取消单个操作

 (2)暂停和恢复队列

复制代码 代码如下:

- (void)setsuspended:(bool)b; // yes代表暂停队列,no代表恢复队列

- (bool)issuspended; //当前状态


(3)暂停和恢复的适用场合:在tableview界面,开线程下载远程的网络界面,对ui会有影响,使用户体验变差。那么这种情况,就可以设置在用户操作ui(如滚动屏幕)的时候,暂停队列(不是取消队列),停止滚动的时候,恢复队列。

五、操作优先级
 (1)设置nsoperation在queue中的优先级,可以改变操作的执⾏优先级

复制代码 代码如下:

- (nsoperationqueuepriority)queuepriority;
- (void)setqueuepriority:(nsoperationqueuepriority)p;

 (2)优先级的取值
复制代码 代码如下:

nsoperationqueuepriorityverylow = -8l,

nsoperationqueueprioritylow = -4l,

nsoperationqueueprioritynormal = 0,

nsoperationqueuepriorityhigh = 4,

nsoperationqueuepriorityveryhigh = 8


说明:优先级高的任务,调用的几率会更大。

六、操作依赖
(1)nsoperation之间可以设置依赖来保证执行顺序,⽐如一定要让操作a执行完后,才能执行操作b,可以像下面这么写

复制代码 代码如下:

[operationb adddependency:operationa]; // 操作b依赖于操作

(2)可以在不同queue的nsoperation之间创建依赖关系

iOS多线程应用开发中使用NSOperation类的基本方法

注意:不能循环依赖(不能a依赖于b,b又依赖于a)。

(3)代码示例

复制代码 代码如下:

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];

    //创建nsinvocationoperation对象,封装操作
    nsinvocationoperation *operation1=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test1) object:nil];
    nsinvocationoperation *operation2=[[nsinvocationoperation alloc]initwithtarget:self selector:@selector(test2) object:nil];
    //创建对象,封装操作
    nsblockoperation *operation3=[nsblockoperation blockoperationwithblock:^{
        for (int i=0; i<5; i++) {
            nslog(@"nsblockoperation3--1----%@",[nsthread currentthread]);
        }
    }];
    [operation3 addexecutionblock:^{
        for (int i=0; i<5; i++) {
        nslog(@"nsblockoperation3--2----%@",[nsthread currentthread]);
        }
    }];
   
    //设置操作依赖
    //先执行operation2,再执行operation1,最后执行operation3
    [operation3 adddependency:operation1];
    [operation1 adddependency:operation2];
   
    //不能是相互依赖
//    [operation3 adddependency:operation1];
//    [operation1 adddependency:operation3];
   
    //创建nsoperationqueue
    nsoperationqueue * queue=[[nsoperationqueue alloc]init];
    //把操作添加到队列中
    [queue addoperation:operation1];
    [queue addoperation:operation2];
    [queue addoperation:operation3];
}


复制代码 代码如下:

-(void)test1
{
    for (int i=0; i<5; i++) {
    nslog(@"nsinvocationoperation--test1--%@",[nsthread currentthread]);
    }
}

-(void)test2
{
    for (int i=0; i<5; i++) {
    nslog(@"nsinvocationoperation--test2--%@",[nsthread currentthread]);
    }
}

@end


打印查看:

iOS多线程应用开发中使用NSOperation类的基本方法

a做完再做b,b做完才做c。
注意:一定要在添加之前,进行设置。
提示:任务添加的顺序并不能够决定执行顺序,执行的顺序取决于依赖。使用operation的目的就是为了让开发人员不再关心线程。
 
 
5.操作的监听
可以监听一个操作的执行完毕

复制代码 代码如下:

- (void (^)(void))completionblock;
- (void)setcompletionblock:(void (^)(void))block;

代码示例

第一种方式:可以直接跟在任务后面编写需要完成的操作,如这里在下载图片后,紧跟着下载第二张图片。但是这种写法有的时候把两个不相关的操作写到了一个代码块中,代码的可阅读性不强。

复制代码 代码如下:

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];

    //创建对象,封装操作
    nsblockoperation *operation=[nsblockoperation blockoperationwithblock:^{
        nslog(@"-operation-下载图片-%@",[nsthread currentthread]);
        //.....下载图片后继续进行的操作
        nslog(@"--接着下载第二张图片--");
    }];
    
    //创建队列
    nsoperationqueue *queue=[[nsoperationqueue alloc]init];
    //把任务添加到队列中(自动执行,自动开线程)
    [queue addoperation:operation];
}

@end


第二种方式:
复制代码 代码如下:

#import "yyviewcontroller.h"

@interface yyviewcontroller ()

@end


复制代码 代码如下:

@implementation yyviewcontroller

- (void)viewdidload
{
    [super viewdidload];

    //创建对象,封装操作
    nsblockoperation *operation=[nsblockoperation blockoperationwithblock:^{
        for (int i=0; i<10; i++) {
            nslog(@"-operation-下载图片-%@",[nsthread currentthread]);
        }
    }];
   
    //监听操作的执行完毕
    operation.completionblock=^{
        //.....下载图片后继续进行的操作
        nslog(@"--接着下载第二张图片--");
    };
   
    //创建队列
    nsoperationqueue *queue=[[nsoperationqueue alloc]init];
    //把任务添加到队列中(自动执行,自动开线程)
    [queue addoperation:operation];
}

@end


打印查看:

iOS多线程应用开发中使用NSOperation类的基本方法

说明:在上一个任务执行完后,会执行operation.completionblock=^{}代码段,且是在当前线程执行(2)。