>

封装了需要并发运行的代码,sbf282.com一、NSOper

- 编辑:澳门博发娱乐官网 -

封装了需要并发运行的代码,sbf282.com一、NSOper

一、NSOperation简介

** 简单说明**

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

NSOperation和NSOperationQueue实现多线程的具体步骤:

先将需要执行的操作封装到一个NSOperation对象中

然后将NSOperation对象添加到NSOperationQueue中

系统会⾃动将NSOperationQueue中的NSOperation取出来

将取出的NSOperation封装的操作放到⼀条新线程中执⾏

** NSOperation的子类 **

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

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

NSInvocationOperation

NSBlockOperation

自定义子类继承NSOperation,实现内部相应的⽅法

 

本文并非最终版本,如有更新或更正会第一时间置顶,联系方式详见文末

本文并非最终版本,如有更新或更正会第一时间置顶,联系方式详见文末

NSOperation是苹果提供给我们的一套多线程解决方案。实际上它是在ios4后基于GCD开发的,但是比GCD拥有更强的可控性和代码可读性,并且可以加入操作依赖。

二、 具体说明

创建对象和执行操作:

- viewDidLoad{ [super viewDidLoad]; //NSOperation:抽象类,不具备封装功能 //创建操作对象,封装要执行的任务 //NSInvocationOperation 封装操作 NSInvocationOperation *operation=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector object:nil]; //执行操作 [operation start];}

输出结果如下:

2016-08-22 16:15:44.861 NSOPeration[1928:254446] --test--<NSThread: 0x7f89b3602240>{number = 1, name = main}--

说明: 一旦执⾏操作,就会调用target的test方法操作对象默认在主线程中执行,只有添加到队列中才会开启新的线程。即默认情况下,如果操作没有放到队列中queue中,都是同步执行。只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作 。

创建对象和添加操作:

- 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];}

输出结果如下:

2016-08-22 16:14:09.263 NSOPeration[1915:252851] NSBlockOperation------<NSThread: 0x7fc9b0e01530>{number = 1, name = main}2016-08-22 16:14:09.263 NSOPeration[1915:252997] NSBlockOperation1------<NSThread: 0x7fc9b0d18c70>{number = 3, name = }2016-08-22 16:14:09.263 NSOPeration[1915:252891] NSBlockOperation2------<NSThread: 0x7fc9b0f26370>{number = 2, name = }

说明:只要NSBlockOperation封装的操作数 > 1,就会异步执行操作

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

如果将NSOperation添加到NSOperationQueue中,系统会自动异步执行NSOperation中的操作

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

- viewDidLoad{ [super viewDidLoad]; //创建NSInvocationOperation对象,封装操作 NSInvocationOperation *operation1=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector object:nil]; NSInvocationOperation *operation2=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector 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];}

打印输出:

2016-08-22 16:13:16.773 NSOPeration[1905:251784] NSInvocationOperation--test2--<NSThread: 0x7fee73e2acf0>{number = 2, name = }2016-08-22 16:13:16.773 NSOPeration[1905:251794] NSBlockOperation3--1----<NSThread: 0x7fee73f218b0>{number = 5, name = }2016-08-22 16:13:16.773 NSOPeration[1905:251790] NSInvocationOperation--test1--<NSThread: 0x7fee73d0bbf0>{number = 3, name = }2016-08-22 16:13:16.773 NSOPeration[1905:251826] NSBlockOperation3--2----<NSThread: 0x7fee73e123c0>{number = 4, name = }

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

** 提示:**队列的取出是有顺序的,与打印结果并不矛盾。这就好比,选手A,BC虽然起跑的顺序是先A,后B,然后C,但是到达终点的顺序却不一定是A,B在前,C在后。

NSOperation

如果觉得本文内容过长,请前往本人"简书"

本文源码 Demo 详见 Github

如果觉得本文内容过长,请前往本人"简书"

本文源码 Demo 详见 Github

NSOperation实质是封装了需要并发运行的代码,一些主要接口和NSThread基本相同,可以看做没有线程运行能力的thread类的抽象。

三、NSOperation基本操作

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

- (NSInteger)maxConcurrentOperationCount;- setMaxConcurrentOperationCount:(NSInteger)cnt; 

说明:如果没有设置最大并发数,那么并发的个数是由系统内存和CPU决定的,可能内存多久开多一点,内存少就开少一点。

** 注意:num的值并不代表线程的个数,仅仅代表线程的ID。** 提示:最大并发数不要乱写,不要开太多,一般以2~3为宜,因为虽然任务是在子线程进行处理的,但是cpu处理这些过多的子线程可能会影响UI,让UI变卡。

** 二、队列的取消,暂停和恢复 **取消队列的所有操作

- cancelAllOperations;提⽰:也可以调用NSOperation的- cancel⽅法取消单个操作

暂停和恢复队列

- setSuspended:b; // YES代表暂停队列,NO代表恢复队列- isSuspended; //当前状态

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

** 三、操作优先级 **设置NSOperation在queue中的优先级,可以改变操作的执⾏优先级

- (NSOperationQueuePriority)queuePriority;- setQueuePriority:(NSOperationQueuePriority)p;

优先级的取值,优先级高的任务,调用的几率会更大。

NSOperationQueuePriorityVeryLow = -8L,NSOperationQueuePriorityLow = -4L,NSOperationQueuePriorityNormal = 0,NSOperationQueuePriorityHigh = 4,NSOperationQueuePriorityVeryHigh = 8 

** 四、操作依赖 **NSOperation之间可以设置依赖来保证执行顺序,⽐如一定要让操作A执行完后,才能执行操作B,可以像下面这么写

[operationB addDependency:operationA]; // 操作B依赖于操作

可以在不同queue的NSOperation之间创建依赖关系

sbf282.com 1依赖关系

注意:不能循环依赖(不能A依赖于B,B又依赖于A)。

代码示例

- viewDidLoad{ [super viewDidLoad]; //创建NSInvocationOperation对象,封装操作 NSInvocationOperation *operation1=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector object:nil]; NSInvocationOperation *operation2=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector 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];}-test1{ for (int i=0; i<5; i++) { NSLog(@"NSInvocationOperation--test1--%@",[NSThread currentThread]); }}-test2{ for (int i=0; i<5; i++) { NSLog(@"NSInvocationOperation--test2--%@",[NSThread currentThread]); }}

输出结果:

2016-08-22 16:52:14.719 NSOPeration[2102:274395] NSInvocationOperation--test2--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.728 NSOPeration[2102:274395] NSInvocationOperation--test2--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.758 NSOPeration[2102:274395] NSInvocationOperation--test2--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.759 NSOPeration[2102:274395] NSInvocationOperation--test1--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.782 NSOPeration[2102:274395] NSInvocationOperation--test1--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.784 NSOPeration[2102:274395] NSInvocationOperation--test1--<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.785 NSOPeration[2102:274395] NSBlockOperation3--1----<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.785 NSOPeration[2102:274394] NSBlockOperation3--2----<NSThread: 0x7f962051f510>{number = 3, name = }2016-08-22 16:52:14.785 NSOPeration[2102:274395] NSBlockOperation3--1----<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.785 NSOPeration[2102:274394] NSBlockOperation3--2----<NSThread: 0x7f962051f510>{number = 3, name = }2016-08-22 16:52:14.786 NSOPeration[2102:274395] NSBlockOperation3--1----<NSThread: 0x7f962040e970>{number = 2, name = }2016-08-22 16:52:14.794 NSOPeration[2102:274394] NSBlockOperation3--2----<NSThread: 0x7f962051f510>{number = 3, name = }

A做完再做B,B做完才做C。

** 注意:一定要在添加之前,进行设置。** 提示:任务添加的顺序并不能够决定执行顺序,执行的顺序取决于依赖。使用Operation的目的就是为了让开发人员不再关心线程。

** 5.操作的监听 **可以监听一个操作的执行完毕

- completionBlock;- setCompletionBlock:block; 

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

- viewDidLoad{ [super viewDidLoad]; //创建对象,封装操作 NSBlockOperation *operation=[NSBlockOperation blockOperationWithBlock:^{ NSLog(@"-operation-下载图片-%@",[NSThread currentThread]); //.....下载图片后继续进行的操作 NSLog(@"--接着下载第二张图片--"); }]; //创建队列 NSOperationQueue *queue=[[NSOperationQueue alloc]init]; //把任务添加到队列中(自动执行,自动开线程) [queue addOperation:operation];}

第二种方式 代码示例:

- 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];}

输出结果如下:

2016-08-22 16:56:19.397 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.398 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.399 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.399 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.400 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.400 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.401 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.401 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.402 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.402 NSOPeration[2125:278164] -operation-下载图片-<NSThread: 0x7fde73608080>{number = 3, name = }2016-08-22 16:56:19.403 NSOPeration[2125:278156] --接着下载第二张图片--

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

一、NSOperation简单介绍

1.0 NSOperation 的作用

使用 NSOperation 的目的就是为了让开发人员不再关心线程

  • 配合使用 NSOperation(任务) 和 NSOperationQueue(队列) 也能实现多线程编程

1.0 NSOperation 的作用

使用 NSOperation 的目的就是为了让开发人员不再关心线程

  • 配合使用 NSOperation(任务) 和 NSOperationQueue(队列) 也能实现多线程编程

一、多线程NSOperation简介

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 *invocationOperation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(loadImageSource:) object:imgUrl];
  [invocationOperation start];//直接会在当前线程执行,start调用之后会直接执行loadImageSource:方法

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

NSInvocationOperation *invocationOperation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(loadImageSource:) object:imgUrl];
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[queue addOperation:invocationOperation];

2.NSBlockOperation子类
创建对象和执行操作:

 NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
    [self loadImageSource:imgUrl];
}];
//开启执行操作
[blockOperation start];

NSBlockOperation *blockOperation1 = [[NSBlockOperation alloc] init];
[blockOperation1 addExecutionBlock:^{
    [self loadImageSource:imgUrl];
}];
[blockOperation1 addExecutionBlock:^{
    NSLog(@"NSBlockOperation1------%@",[NSThread currentThread]);
}];
 //开启执行操作
[blockOperation1  start];

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

NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
    [self loadImageSource:imgUrl];
}];

NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[queue addOperation:blockOperation];

3.NSOperationQueue

NSOperationQueue的作⽤:NSOperation可以调⽤start⽅法来执⾏任务,但默认是同步执行的
如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动异步执行NSOperation中的操作
添加操作到NSOperationQueue中,自动执行操作,自动开启线程
添加方法有三个:

  • (void)addOperation:(NSOperation *)op;
  • (void)addOperationWithBlock:(void (^)(void))block;
    // waitUntilFinished NO 异步 YES 同步
  • (void)addOperations:(NSArray<NSOperation *> *)ops waitUntilFinished:(BOOL)wait;

、自定义NSOperation

个人觉得没必要,要自定义,对异步处理网络加载、图片下载的有AFNetWorking、SDWebImage第三方框架。

原文链接

1、NSOperation简介

NSOperation 和 NSOperationQueue 实现多线程的具体步骤:

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

(2)然后将NSOperation对象添加到NSOperationQueue中

(3)系统会自动将NSOperationQueue中的NSOperation取出来

(4)将取出的NSOperation封装的操作放到一条新线程中执行

NSOperation 和 NSOperationQueue 实现多线程的具体步骤:

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

(2)然后将NSOperation对象添加到NSOperationQueue中

(3)系统会自动将NSOperationQueue中的NSOperation取出来

(4)将取出的NSOperation封装的操作放到一条新线程中执行

二、NSOperationQueue的其他属性

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

NSOperationQueue * queue=[[NSOperationQueue alloc]init];
queue.maxConcurrentOperationCount = 2;
[queue setMaxConcurrentOperationCount: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; //当前状态

      queue.suspended = !queue.isSuspended;

暂停任务:未开始的任务和已经开始但还没有执行完的任务(点击暂停后这些会执行完)

如果队列中没有任务了,那就不用suspended属性了,需要做判断

if (queue.operationCount == 0) {
    NSLog(@"队列中的任务已经被清空");
    return;
}

(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之间创建依赖关系
注意:不能循环依赖(不能A依赖于B,B又依赖于A)。

(3)注意:必须在添加到队列前,设置依赖关系
任务添加的顺序并不能够决定执行顺序,执行的顺序取决于依赖。使用Operation的目的就是为了让开发人员不再关心线程。

五、操作的监听
可以监听一个操作的执行完毕

  • (void (^)(void))completionBlock;
  • (void)setCompletionBlock:(void (^)(void))block;

①.简单说明

  • NSOperation的作⽤:配合使用NSOperation和NSOperationQueue也能实现多线程编程

使用NSOperation子类的方式有3种:

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

  1. NSInvocationOperation
  • NSBlockOperation
  • 自定义子类继承NSOperation,实现内部相应的方法

使用NSOperation子类的方式有3种:

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

  1. NSInvocationOperation
  • NSBlockOperation
  • 自定义子类继承NSOperation,实现内部相应的方法

三、自定义NSOperation

自定义NSOperation的步骤很简单
重写-(void)main方法,在里面实现想执行的任务

具体的使用可参照
https://github.com/zhangyanxiao/MutiTreadDemo
中的NSOperation部分

NSOperation和NSOperationQueue实现多线程的具体步骤:
  • (1)先将需要执行的操作封装到一个NSOperation对象中

  • (2)然后将NSOperation对象添加到NSOperationQueue中

  • (3)系统会⾃动将NSOperationQueue中的NSOperation取出来

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

2.0 NSInvocationOperation

//创建NSInvocationOperation对象
- (id)initWithTarget:(id)target selector:(SEL)sel object:(id)arg;

//调用start方法开始执行操作,一旦执行操作,就会调用target的sel方法
- (void)start;

注意:

  • 默认情况下,操作对象在主线程中执行
  • 调用了start方法后并不会开一条新线程去执行操作,只有添加到队列中才会开启新的线程
  • 即默认情况下,如果操作没有放到队列中queue中,都是同步执行。
  • 只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作

代码示例:

#import "ViewController.h"

@interface ViewController ()
@end

@implementation ViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

  //创建操作对象,封装要执行的任务
  NSInvocationOperation *op =
      [[NSInvocationOperation alloc] initWithTarget:self
                                           selector:@selector(run)
                                             object:nil];
  //执行操作
  [op start];
}

- (void)run {
  NSLog(@"------%@", [NSThread currentThread]);
}

- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
  // Dispose of any resources that can be recreated.
}
@end

打印结果:

NSInvocationOperation[862:29437] ------<NSThread: 0x7f9cea507920>{number = 1, name = main}

2.0 NSInvocationOperation

//创建NSInvocationOperation对象
- (id)initWithTarget:(id)target selector:(SEL)sel object:(id)arg;

//调用start方法开始执行操作,一旦执行操作,就会调用target的sel方法
- (void)start;

注意:

  • 默认情况下,操作对象在主线程中执行
  • 调用了start方法后并不会开一条新线程去执行操作,只有添加到队列中才会开启新的线程
  • 即默认情况下,如果操作没有放到队列中queue中,都是同步执行。
  • 只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作

代码示例:

#import "ViewController.h"

@interface ViewController ()
@end

@implementation ViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

  //创建操作对象,封装要执行的任务
  NSInvocationOperation *op =
      [[NSInvocationOperation alloc] initWithTarget:self
                                           selector:@selector(run)
                                             object:nil];
  //执行操作
  [op start];
}

- (void)run {
  NSLog(@"------%@", [NSThread currentThread]);
}

- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
  // Dispose of any resources that can be recreated.
}
@end

打印结果:

NSInvocationOperation[862:29437] ------<NSThread: 0x7f9cea507920>{number = 1, name = main}

②.NSOperation的子类

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

3.0 NSBlockOperation

//创建 NSBlockOperation 操作对象
+ (id)blockOperationWithBlock:(void (^)(void))block;

// 添加操作
- (void)addExecutionBlock:(void (^)(void))block;

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


代码示例:

#import "ViewController.h"

@interface ViewController ()
@end

@implementation ViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

  // 1.创建 NSBlockOperation 操作对象
  NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
    // 在主线程
    NSLog(@"下载1------%@", [NSThread currentThread]);
  }];

  // 2.添加操作(额外的任务)(在子线程执行)
  [op addExecutionBlock:^{
    NSLog(@"下载2------%@", [NSThread currentThread]);
  }];

  [op addExecutionBlock:^{

  [op addExecutionBlock:^{
    NSLog(@"下载2------%@", [NSThread currentThread]);
  }];
  [op addExecutionBlock:^{
    NSLog(@"下载3------%@", [NSThread currentThread]);
  }];
  [op addExecutionBlock:^{
    NSLog(@"下载4------%@", [NSThread currentThread]);
  }];
  // 3.开启执行操作
  [op start];
}
- (void)run {
  NSLog(@"------%@", [NSThread currentThread]);
}
- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
  // Dispose of any resources that can be recreated.
}
@end

打印结果:

NSBlockOperation[1013:37922] 下载1------<NSThread: 0x7feea1c05460>{number = 1, name = main}
NSBlockOperation[1013:37952] 下载2------<NSThread: 0x7feea1f0b790>{number = 2, name = (null)}
NSBlockOperation[1013:37955] 下载3------<NSThread: 0x7feea1c0f8a0>{number = 3, name = (null)}
NSBlockOperation[1013:37951] 下载4------<NSThread: 0x7feea1e0b520>{number = 4, name = (null)}

3.0 NSBlockOperation

//创建 NSBlockOperation 操作对象
+ (id)blockOperationWithBlock:(void (^)(void))block;

// 添加操作
- (void)addExecutionBlock:(void (^)(void))block;

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


代码示例:

#import "ViewController.h"

@interface ViewController ()
@end

@implementation ViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

  // 1.创建 NSBlockOperation 操作对象
  NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
    // 在主线程
    NSLog(@"下载1------%@", [NSThread currentThread]);
  }];

  // 2.添加操作(额外的任务)(在子线程执行)
  [op addExecutionBlock:^{
    NSLog(@"下载2------%@", [NSThread currentThread]);
  }];

  [op addExecutionBlock:^{

  [op addExecutionBlock:^{
    NSLog(@"下载2------%@", [NSThread currentThread]);
  }];
  [op addExecutionBlock:^{
    NSLog(@"下载3------%@", [NSThread currentThread]);
  }];
  [op addExecutionBlock:^{
    NSLog(@"下载4------%@", [NSThread currentThread]);
  }];
  // 3.开启执行操作
  [op start];
}
- (void)run {
  NSLog(@"------%@", [NSThread currentThread]);
}
- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
  // Dispose of any resources that can be recreated.
}
@end

打印结果:

NSBlockOperation[1013:37922] 下载1------<NSThread: 0x7feea1c05460>{number = 1, name = main}
NSBlockOperation[1013:37952] 下载2------<NSThread: 0x7feea1f0b790>{number = 2, name = (null)}
NSBlockOperation[1013:37955] 下载3------<NSThread: 0x7feea1c0f8a0>{number = 3, name = (null)}
NSBlockOperation[1013:37951] 下载4------<NSThread: 0x7feea1e0b520>{number = 4, name = (null)}

本文由胜博发-编程发布,转载请注明来源:封装了需要并发运行的代码,sbf282.com一、NSOper