NSOperation是基于GCD的一套多线程实现方案,和GCD一样,线程的生命周期是由系统来自动管理的,不用像NSThread和Pthread一样让程序员手动管理。相对于GCD来说,它更加地面向对象,并且比GCD多了一些更加简单实用的功能,另外,由于它的API是纯OC的,深受广大程序员喜爱,实用频率很高。

NSOperation主要和NSOperationQueue配合使用实现多线程,一般步骤如下:

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

2.然后将NSOperation对象添加到NSOperationQueue中;

3.系统会自动将NSOperationQueue中的NSOperation取出来;

4.将取出的NSOperation放到一条线程中执行。

注意:NSOperation是一个抽象类,并不具备封装操作的能力,必须使用它的子类。使用NSOperation子类的方式有如下三种:

1.NSInvocationOperation

2.NSBlockOperation

3.自定义子类继承NSOperation,实现内部相应的方法。

我们首先来看前两种:

一、NSOperation:

 - (void)viewDidLoad {
[super viewDidLoad];
[self invocationOperation];
} -(void)invocationOperation
{ // 1.创建操作对象,封住需要执行的任务
NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download1) object:nil]; // 2.执行操作(默认情况下,如果操作没有放到操作队列中,同步执行)
[operation start];
} -(void)download1
{
NSLog(@"下载1----%@",[NSThread currentThread]);
}

结果如下:

可以看出,默认情况下,如果操作没有放到操作队列中,会在主线程同步执行,只有讲NSOperation放到NSOperationQueue中才会异步执行。

二、NSBlockOperation

 - (void)viewDidLoad {
[super viewDidLoad];
[self blockOperation];
} - (void)blockOperation
{
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"NSBlockOperation--下载1---%@",[NSThread currentThread]);
}]; [operation addExecutionBlock:^{
NSLog(@"NSBlockOperation--下载2---%@",[NSThread currentThread]);
}]; [operation addExecutionBlock:^{
NSLog(@"NSBlockOperation--下载3---%@",[NSThread currentThread]);
}];
[operation start];
}

结果如图:

我们会发现,NSBlockOperation当只有单个任务的时候,也是默认在主线程执行,当任务数大于1的时候,会开启子线程并发执行其它的操作

三、配合使用NSOperationQueue

 - (void)viewDidLoad {
[super viewDidLoad];
[self operationQueue];
} - (void)operationQueue
{
NSInvocationOperation *operation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download1) object:nil];
NSInvocationOperation *operation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download2) object:nil];
NSInvocationOperation *operation3 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download3) object:nil];
NSBlockOperation *operation4 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"NSBlockOperation--1---%@",[NSThread currentThread]);
}];
[operation4 addExecutionBlock:^{
NSLog(@"NSBlockOperation--2---%@",[NSThread currentThread]);
}];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[queue addOperation:operation1];
[queue addOperation:operation2];
[queue addOperation:operation3];
[queue addOperation:operation4];
}
-(void)download1
{
NSLog(@"下载1----%@",[NSThread currentThread]);
}
-(void)download2
{
NSLog(@"下载2----%@",[NSThread currentThread]);
}
-(void)download3
{
NSLog(@"下载3----%@",[NSThread currentThread]);
}

结果如下

可以看出,只要是添加到NSOperationQueue中的操作,系统都会自动为我们开启子线程来执行,并且是并发无序的执行,和添加的顺序无关。

然而有些时候我们需要明确地指定操作的顺序,我们可以在NSOperation之间设置依赖来保证执行顺序。

比如操作1要在操作2后面执行,可以这么写:[operation1 addDependency:operation2] 意思是operation1依赖于operation2,也就是operation1要在operation2后面执行

上代码验证下:

 - (void)viewDidLoad {
[super viewDidLoad];
[self operationQueue];
} - (void)operationQueue
{
NSInvocationOperation *operation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download1) object:nil];
NSInvocationOperation *operation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download2) object:nil];
NSInvocationOperation *operation3 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download3) object:nil];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[operation3 addDependency:operation1];
[operation1 addDependency:operation2];
[queue addOperation:operation1];
[queue addOperation:operation2];
[queue addOperation:operation3];
}
-(void)download1
{
NSLog(@"下载1----%@",[NSThread currentThread]);
}
-(void)download2
{
NSLog(@"下载2----%@",[NSThread currentThread]);
}
-(void)download3
{
NSLog(@"下载3----%@",[NSThread currentThread]);
}

结果如下:

操作顺序Operation2->Operation1->Operation3,完全正确,其实就是谁依赖谁,谁就在它的后面执行

另外也可以在不同queue的NSOperation之间设置依赖

操作顺序:4->3->2->1 其余两个并发执行

下面列出了NSOperationQueue的一些属性和方法:

1.- (void)cancelAllOperations 取消队列的所有操作。PS:也可以调用NSOperation的-(void)cancel方法取消单个操作

2.@property (getter=isSuspended) BOOL suspended;YES表示暂停队列,NO表示回复队列

3.@property NSInteger maxConcurrentOperationCount;表示最大并发数,一般不大于5

四、自定义NSOperation

当将自定义的NSOperation添加到NSOperationQueue的时候,系统会调用NSOperation的-(void)main方法,所以我们只要重写这个方法实现相应的实现即可。

 -(void)main
{
@autoreleasepool { // 异步线程无法访问主线程的自动释放池,需要自己添加
if (self.isCancelled) return;// 时刻监听操作是否取消,若取消就返回
NSURL *url = [NSURL URLWithString:self.matchUrl];
if (self.isCancelled) return;
NSData *data = [NSData dataWithContentsOfURL:url];
if (self.isCancelled) return;
UIImage *image = [UIImage imageWithData:data];
if ([_delegate respondsToSelector:@selector(operation:finshedDownloadImage:)]) {
dispatch_async(dispatch_get_main_queue(), ^{// 在主线程更新UI
[self.delegate operation:self finshedDownloadImage:image];
});
}
}
}

注意由于子线称无法访问主线程的自动释放池,所以需要自己添加。

下面是个小Demo截图,自定义NSOperation实现图片的异步下载,以及避免重复下载同一张图片

iOS多线程--NSOperation的更多相关文章

  1. iOS 多线程 NSOperation、NSOperationQueue

    1. NSOperation.NSOperationQueue 简介 NSOperation.NSOperationQueue 是苹果提供给我们的一套多线程解决方案.实际上 NSOperation.N ...

  2. iOS多线程 NSOperation的用法

    上一篇写了 GCD 的使用,接下来就了解一下 NSOperation ,NSOperation是苹果对 GCD 的 OC 版的一个封装,但是相对于GCD来说可控性更强,并且可以加入操作依赖. NSOp ...

  3. iOS多线程--NSOperation 浅显易懂

    NSOperation是基于GCD的一套多线程实现方案,和GCD一样,线程的生命周期是由系统来自动管理的,不用像NSThread和Pthread一样让程序员手动管理.相对于GCD来说,它更加地面向对象 ...

  4. iOS多线程---NSOperation介绍和使用

    1.  NSOperation实现多线程编程,需要和NSOperationQueue一起使用. (1)先将要执行的操作封装到NSOperation中 (2)将NSOperation对象添加到NSOpe ...

  5. IOS 多线程 NSOperation GCD

    1.NSInvocationOperation NSInvocationOperation * op; NSOperationQueue * que = [[NSOperationQueuealloc ...

  6. iOS多线程---NSOperation的常用操作

    1.最大并发数: - (NSInteger)maxConcurrentOperationCount;- (void)setMaxConcurrentOperationCount:(NSInteger) ...

  7. iOS多线程编程

    废话不多说,直接上干货.先熟悉一下基本知识,然后讲一下常用的两种,NSOperation和GCD. 一.基础概念 进程: 狭义定义:进程是正在运行的程序的实例(an instance of a com ...

  8. iOS GCD NSOperation NSThread等多线程各种举例详解(拷贝)

    2年多的iOS之路匆匆而过,期间也拜读来不少大神的博客,近来突然为自己一直做伸手党感到羞耻,是时候回馈社会.回想当年自己还是小白的时候,照着一些iOS多线程教程学,也只是照抄,只知其然.不知其所以然. ...

  9. iOS 多线程学习笔记 —— NSOperation

    本文复制.参考自文章:iOS多线程编程之NSOperation和NSOperationQueue的使用 ,主要为了加强个人对知识的理解和记忆,不做他用.原作者声明: 著作权声明:本文由http://b ...

随机推荐

  1. 2015年10月22日CSS学习笔记

    XHTML1.0对HTML4.0的改进 借鉴了XML的写法,语法更加严格. 把页面的内容和样式分离了,废弃了html4中的表示样式的标签和属性.推荐使用css来描述页面的样式. CSS样式的优先级 ! ...

  2. 全面产品管理-从细微处认识"用户体验"

    转载: 让我以一个故事开始本文,我觉得这个故事能概括大多数人听到“用户体验”这个术语时的想法. 我经常访问的一个财经网站给我发了一封电子邮件,请求我点击里面的一个链接,对一些信息进行审核.所以我就点了 ...

  3. sql执行计划

    http://www.cnblogs.com/kissdodog/p/3160560.html

  4. Eclipse创建Maven Web项目 + 测试覆盖率 + 常见问题(2015.07.14——湛耀)

    Eclipse创建Maven web项目: 到此,并没有创建好,接下来一步步解决问题: 问题:无法创建src/main/java目录 解决: 右键项目选择[properties] 点击[OK] 问题: ...

  5. 为静态Checkbox动态地添加checked属性

    1.ASP.NET HTML Code: 嵌套在repeater中 " ? "checked" : "" %> /> *** 关键代码: ...

  6. Yii Active Record 查询结果转化成数组

    使用Yii 的Active Record 来获取查询结果的时候,返回的结果集是一个对象类型的,有时候为了数据处理的方便希望能够转成数组返回.比如下面的方法: // 查找满足指定条件的结果中的第一行 $ ...

  7. key_t键和ftok函数

    系统建立IPC通讯(如消息队列.共享内存时)必须指定一个ID值.通常情况下,该id值通过ftok函数得到. ftok原型如下: key_t ftok( char * fname, int id ) f ...

  8. 转储指定的数据块并查看TRC信息

    1.转储指定的块:需要两个信息:文件号和块号 BYS@bys1>alter system dump datafile 1 block 100; System altered. 2.定位找出use ...

  9. JS跨域笔记

    什么是跨域,跨域是指不同域之间相互访问,只要协议.域名.端口有任何一个不同,都被当作是不同的域. 对于端口和协议的不同,只能通过后台来解决,前台是无能为力的. 受浏览器同源策略的限制,本域的js不能操 ...

  10. JSF学习四 标签

    commandButton:提交.重置或下压button button:用于公布GET请求的按钮