iOS多线程之8.NSOPeration的其他用法
本文主要对NSOPeration的一些重点属性和方法做出介绍,以便大家可以更好的使用NSOPeration。
1.添加依赖
- (void)addDependency:(NSOperation *)op;
需求:同时下载两张图片,两张图片都下载完了,在合成成一张。这个例子我在iOS多线程之6.GCD的其他用法这篇文章中用过,当时是用GCD的group实现的。这次我们用NSOPeration实现。
代码
// 点击屏幕下载图片
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
__block UIImage *image1 = nil;
// 下载图片1
NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"下载第一张图片%@",[NSThread currentThread]);
NSString *strURL1 = @"http://h.hiphotos.baidu.com/zhidao/pic/item/6d81800a19d8bc3ed69473cb848ba61ea8d34516.jpg";
image1 = [self downloadImageWithURL:strURL1];
}];
__block UIImage *image2 = nil;
// 下载图片2
NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"下载第二张图片%@",[NSThread currentThread]);
NSString *strURL2 = @"http://h.hiphotos.baidu.com/zhidao/pic/item/0eb30f2442a7d9334f268ca9a84bd11372f00159.jpg";
image2 = [self downloadImageWithURL:strURL2];
}];
// 两张图片下载完 再合并图片
NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"合并图片%@",[NSThread currentThread]);
// 在主线程刷新UI
dispatch_async(dispatch_get_main_queue(), ^{
self.imageView1.image = image1;
self.imageView2.image = image2;
// 合并两张图片图片
UIGraphicsBeginImageContextWithOptions(CGSizeMake(200, 100), NO, 0.0);
[image1 drawInRect:CGRectMake(0, 0, 100, 100)];
[image2 drawInRect:CGRectMake(100, 0, 100, 100)];
self.imageView3.image = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
});
}];
// 添加依赖
[operation3 addDependency:operation1];
[operation3 addDependency:operation2];
// 创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
// 把操作放队列中
[queue addOperation:operation1];
[queue addOperation:operation2];
[queue addOperation:operation3];
}
- (UIImage *)downloadImageWithURL : (NSString *)strURL {
NSURL *url = [NSURL URLWithString:strURL];
return [UIImage imageWithData:[NSData dataWithContentsOfURL:url]];
}
日志
2016-11-12 09:23:42.013 TTTTTTTTTT[2544:33281] 下载第二张图片<NSThread: 0x60000007ec40>{number = 4, name = (null)}
2016-11-12 09:23:42.013 TTTTTTTTTT[2544:33282] 下载第一张图片<NSThread: 0x60000007cac0>{number = 3, name = (null)}
2016-11-12 09:23:42.141 TTTTTTTTTT[2544:33309] 合并图片<NSThread: 0x60800007e700>{number = 6, name = (null)}
效果:
分析 :只有两张图片下载完,合并图片才有意义,所以operation3里面的操作必须等operation1和operation2里操作完成才能执行。
[operation3 addDependency:operation1];
operation3依赖于operation1,就是operation3等operation1执行完再执行。
注意:依赖一定在把operation添加进queue之前添加,否则就没有意义了。
既有添加依赖,就有移除依赖。
- (void)removeDependency:(NSOperation *)op;
2.设置operation的优先级
@property NSOperationQueuePriority queuePriority;
当把操作(operation)放入队列(queue)中的时候,是遵守先进先出原则。但是如果你设置了操作(operation)的优先级,那么优先级高的就可以先执行。NSOperationQueuePriority 是一个枚举,共有五个值。
typedef NS_ENUM(NSInteger, NSOperationQueuePriority) {
NSOperationQueuePriorityVeryLow = -8L,// 优先级很低
NSOperationQueuePriorityLow = -4L,// 优先级低
NSOperationQueuePriorityNormal = 0,// 优先级正常
NSOperationQueuePriorityHigh = 4,// 优先级高
NSOperationQueuePriorityVeryHigh = 8// 优先级很高
};
e.g.:上面的例子,是第二张图片先下载的,是先执行operation2的,如果在把操作放在队列中之前设置operation1 的优先级为很高,就可以做到先下载第一张图片,先执行operation1。
operation1.queuePriority = NSURLSessionTaskPriorityHigh;
日志:
2016-11-12 10:00:04.114 TTTTTTTTTT[4219:55622] 下载第一张图片<NSThread: 0x6080002613c0>{number = 3, name = (null)}
2016-11-12 10:00:04.115 TTTTTTTTTT[4219:55611] 下载第二张图片<NSThread: 0x600000075340>{number = 4, name = (null)}
2016-11-12 10:00:04.267 TTTTTTTTTT[4219:55644] 合并图片<NSThread: 0x600000264900>{number = 6, name = (null)}
注意:优先级高的就一定会最先执行吗?不一定,这个例子比较简单,不能代表全部。优先级高只是提供了一个可能,至于会不会最先执行,还要看CPU的使用情况、操作的复杂程度和队列。同理,优先级低的也不一定会最后执行。
3.操作的取消、执行、完成
NSOperation的三个属性cancelled、executing、finished,分别就是取消,执行,完成。这三个属性都是只读的,我们通过这三个属性可以判断NSOperation的状态,是否取消了,是否正在执行,是否已经完成了。
4.completionBlock
如果你想在操作(operation)完成之后执行一些代码,可以写在这个block块里面。
operation.completionBlock = ^{
NSLog(@"我完成了");
};
注意:如果你把操作(operation)放入队列(queue)里面了,这行代码一定要放在添加之前,否则不执行。
5队列的最大并发数
@property NSInteger maxConcurrentOperationCount;
这是NSOperationQueue的一个属性。由于NSOperationQueue里的操作都是并发的,所以我们可以设置同时并发多少个操作。如果不设置,系统默认。
queue.maxConcurrentOperationCount = 3;
注意:这个属性最好不要设置,系统默认就好。如果设置,最好不要超过5,最好是2-3。如果设置的太大, 会卡顿UI。因为CPU在多个线程之间切换,线程太多,什么时候才能轮到主线程刷新UI啊!。
6.队列的暂停、恢复、取消
1)暂停:
[queue setSuspended:YES];
2)恢复:
[queue setSuspended:NO];
3)判读队列的当前状态:
@property (getter=isSuspended) BOOL suspended;
当你把队列暂停时,队列里的操作就不执行了。当你滑动列表时,可以先把队列(队列里执行下载图片的操作,列表里的cell上有图片)暂停,不滑动时再恢复,增加APP的流畅性。
4)取消所有操作
- (void)cancelAllOperations;
队列里的所有操作都不执行了。
以上就是关于NSOperation的常用操作,用这些基本上就能够满足我们的需求。如果还满足不了怎么办,自定义NSOperation,下一篇文章讲自定义NSOperation。
iOS多线程之8.NSOPeration的其他用法的更多相关文章
- iOS多线程之7.NSOperation的初识
NSOperation和GCD一样,不用我们管理线程的生命周期,加锁等问题,只要把操作封装进NSOperation中,系统会自动帮我们创建线程,执行操作.而且他是面向对象的,我们看起来更容易理解,使用 ...
- iOS多线程之6.GCD的其他用法
队列组 让队列里的任务同时执行,当任务都执行完毕时,再以通知的形式告诉程序员.举例,同时下载两张图片,两张图片都下载完了,在合成成一张. 代码: #import "ViewControl ...
- iOS多线程之GCD小记
iOS多线程之GCD小记 iOS多线程方案简介 从各种资料中了解到,iOS中目前有4套多线程的方案,分别是下列4中: 1.Pthreads 这是一套可以在很多操作系统上通用的多线程API,是基于C语言 ...
- iOS多线程之Thread
多线程 • Thread 是苹果官方提供的,简单已用,可以直接操作线程对象.不过需要程序员自己管理线程的生命周期,主要是创建那部分 优缺点 面向对象,简单易用 直接操作线程对象 需要自己管理线程生命周 ...
- iOS多线程之GCD、OperationQueue 对比和实践记录
[toc] 简介 在计算的早期,计算机可以执行的最大工作量是由 CPU 的时钟速度决定的.但是随着技术的进步和处理器设计的紧凑化,热量和其他物理约束开始限制处理器的最大时钟速度.因此,芯片制 ...
- iOS多线程之NSOperation详解
使用NSOperation和NSOperationQueue进行多线程开发,只要将一个NSOperation(实际开发中需要使用其子类 NSInvocationOperation,NSBlockOpe ...
- iOS 多线程之NSOperation篇举例详解
这篇博客是接着总篇iOS GCD NSOperation NSThread等多线程各种举例详解写的一个支篇.总篇也包含了此文的链接.本文讲解的知识点有NSBlockOperationClick,队列, ...
- iOS多线程之9.自定义NSOperation
本文主要讲如何自定义NSOperation,以及自定义NSOperation的一些注意事项,以下载图片为例. 新建一个类,继承于NSOperation. CustomOperation.h 代码 ...
- IOS多线程之NSOperation学习总结
NSOperation简介 1.NSOperation的作用 配合使用NSOperation和NSOperationQueue也能实现多线程编程 2.NSOperation和NSOperationQu ...
随机推荐
- 【腾讯Bugly干货分享】Android Patch 方案与持续交付
本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57a31921ac3a1fb613dd40f3 Android 不仅系统版本众多 ...
- Leetcode 笔记 113 - Path Sum II
题目链接:Path Sum II | LeetCode OJ Given a binary tree and a sum, find all root-to-leaf paths where each ...
- Egret3D研究报告(二)从Unity3D导出场景到Egret3D
Egret3D暂时没有场编的计划,但是我们知道unity3D是一个很好的场编. 有一些游戏即使不是使用Unity3D开发,也使用Unity3D做场编.这里就不点名了,而且并不在少数. 我们就这么干. ...
- .NET里简易实现IoC
.NET里简易实现IoC 前言 在前面的篇幅中对依赖倒置原则和IoC框架的使用只是做了个简单的介绍,并没有很详细的去演示,可能有的朋友还是区分不了依赖倒置.依赖注入.控制反转这几个名词,或许知道的也只 ...
- 最大子段和(c++)
// 最大子段和.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> using namesp ...
- ABP(现代ASP.NET样板开发框架)系列之9、ABP设置管理
点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之9.ABP设置管理 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)” ...
- Entity Framework 6 Recipes 2nd Edition(10-5)译 -> 在存储模型中使用自定义函数
10-5. 在存储模型中使用自定义函数 问题 想在模型中使用自定义函数,而不是存储过程. 解决方案 假设我们数据库里有成员(members)和他们已经发送的信息(messages) 关系数据表,如Fi ...
- is和as
一.明确两个基本概念 隐式转换: a.对于值类型,低精度=>高精度.eg:int=>long b.对于引用类型,子类向祖宗类转换过程.eg:对象=>Object 显式转换:显示转换是 ...
- js基本类型和引用类型
先来两个例题 //1. var person; person.age=10; console.log(person.age) //undefined person是字符串而不是对象,没有属性 //2. ...
- .NET Core下的日志(3):如何将日志消息输出到控制台上
当我们利用LoggerFactory创建一个Logger对象并利用它来实现日志记录,这个过程会产生一个日志消息,日志消息的流向取决于注册到LoggerFactory之上的LoggerProvider. ...