[iOS 多线程 & 网络 - 1.3] - NSOperation

A.NSOperation的基本使用

1.NSOperation的作用

配合使用NSOperation和NSOperationQueue也能实现多线程编程

NSOperation和NSOperationQueue实现多线程的具体步骤
先将需要执行的操作封装到一个NSOperation对象中
然后将NSOperation对象添加到NSOperationQueue中
系统会自动将NSOperationQueue中的NSOperation取出来
将取出的NSOperation封装的操作放到一条新线程中执行

2.NSOperation的子类

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

使用NSOperation子类的方式有3种
NSInvocationOperation
NSBlockOperation
自定义子类继承NSOperation,实现内部相应的方法

3.NSInvocationOperation

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

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

注意
默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作
只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作

4.NSBlockOperation

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

通过addExecutionBlock:方法添加更多的操作
- (void)addExecutionBlock:(void (^)(void))block;

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

5.NSOperationQueue

NSOperationQueue的作用
NSOperation可以调用start方法来执行任务,但默认是同步执行的
如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动异步执行NSOperation中的操作

添加操作到NSOperationQueue中
- (void)addOperation:(NSOperation *)op;
- (void)addOperationWithBlock:(void (^)(void))block;

6.最大并发数

什么是并发数
同时执行的任务数
比如,同时开3个线程执行3个任务,并发数就是3

最大并发数的相关方法
- (NSInteger)maxConcurrentOperationCount;
- (void)setMaxConcurrentOperationCount:(NSInteger)cnt;

7.队列的取消、暂停和恢复

取消队列的所有操作
- (void)cancelAllOperations;
提示:也可以调用NSOperation的- (void)cancel方法取消单个操作

暂停和恢复队列
- (void)setSuspended:(BOOL)b; // YES代表暂停队列,NO代表恢复队列
- (BOOL)isSuspended;

8.操作优先级

设置NSOperation在queue中的优先级,可以改变操作的执行优先级
- (NSOperationQueuePriority)queuePriority;
- (void)setQueuePriority:(NSOperationQueuePriority)p;

优先级的取值
NSOperationQueuePriorityVeryLow = -8L,
NSOperationQueuePriorityLow = -4L,
NSOperationQueuePriorityNormal = 0,
NSOperationQueuePriorityHigh = 4,
NSOperationQueuePriorityVeryHigh = 8

9.操作依赖

NSOperation之间可以设置依赖来保证执行顺序
比如一定要让操作A执行完后,才能执行操作B,可以这么写
[operationB addDependency:operationA]; // 操作B依赖于操作A

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

10.操作监听

可以监听一个操作的执行完毕
- (void (^)(void))completionBlock;
- (void)setCompletionBlock:(void (^)(void))block;

11.自定义NSOperation

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

重写- (void)main方法的注意点
自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)
经常通过- (BOOL)isCancelled方法检测操作是否被取消,对取消做出响应

B.使用自定义NSOperation后台下载图片

code source: https://github.com/hellovoidworld/ConcurrentDownloadImageDemo

1.思路

2.实现步骤

(1)自定义一个继承NSOperation的类,实现main方法,在main方法中编写任务事件

 1 //
 2 //  HVWDownloadImageOperation.h
 3 //  ConcurrentDownloadImageDemo
 4 //
 5 //  Created by hellovoidworld on 15/1/22.
 6 //  Copyright (c) 2015年 hellovoidworld. All rights reserved.
 7 //
 8
 9 #import <Foundation/Foundation.h>
10
11 @class HVWDownloadImageOperation;
12
13 @protocol HVWDownloadImageOperationDelegate <NSObject>
14
15 /** 代理方法,下载完成之后 */
16 @optional
17 - (void) downloadImageOperation:(HVWDownloadImageOperation *) operation didFinishedDownloadWithImage:(UIImage *) image;
18
19 @end
20
21 @interface HVWDownloadImageOperation : NSOperation
22
23 /** 存储每个图片的url */
24 @property(nonatomic, strong) NSString *url;
25
26 /** 要显示的cell的index */
27 @property(nonatomic, strong) NSIndexPath *indexPath;
28
29 /** 代理 */
30 @property(nonatomic, weak) id<HVWDownloadImageOperationDelegate> delegate;
31
32 @end
 1 //
 2 //  HVWDownloadImageOperation.m
 3 //  ConcurrentDownloadImageDemo
 4 //
 5 //  Created by hellovoidworld on 15/1/22.
 6 //  Copyright (c) 2015年 hellovoidworld. All rights reserved.
 7 //
 8
 9 #import <UIKit/UIKit.h>
10 #import "HVWDownloadImageOperation.h"
11
12
13 @implementation HVWDownloadImageOperation
14
15 - (void)main {
16     NSLog(@"====下载图片======%@", [NSThread currentThread]);
17
18     NSURL *url = [NSURL URLWithString:self.url];
19     NSData *data;
20     for (int i=0; i<1; i++) {
21         data = [NSData dataWithContentsOfURL:url];
22     }
23
24     UIImage *image = [UIImage imageWithData:data];
25
26     /** 调用代理方法,通知代理下载完成 */
27     if ([self.delegate respondsToSelector:@selector(downloadImageOperation:didFinishedDownloadWithImage:)]) {
28         [self.delegate downloadImageOperation:self didFinishedDownloadWithImage:image];
29     }
30 }
31
32 @end

(2)新建一个模型类,用来存储每个cell的数据

 1 //
 2 //  HVWApp.h
 3 //  ConcurrentDownloadImageDemo
 4 //
 5 //  Created by hellovoidworld on 15/1/22.
 6 //  Copyright (c) 2015年 hellovoidworld. All rights reserved.
 7 //
 8
 9 #import <Foundation/Foundation.h>
10
11 @interface HVWApp : NSObject
12
13 /** app名字 */
14 @property(nonatomic, strong) NSString *name;
15 /** app图标url */
16 @property(nonatomic, strong) NSString *icon;
17 /** app的副标题--下载量 */
18 @property(nonatomic, strong) NSString *download;
19
20 - (instancetype) initWithDictionary:(NSDictionary *) dict;
21 + (instancetype) appWithDictionary:(NSDictionary *) dict;
22
23 @end

(3)在控制器中编写调用任务、队列进行多线程并发后台下载图片的操作

  1 //
  2 //  ViewController.m
  3 //  ConcurrentDownloadImageDemo
  4 //
  5 //  Created by hellovoidworld on 15/1/22.
  6 //  Copyright (c) 2015年 hellovoidworld. All rights reserved.
  7 //
  8
  9 #import "ViewController.h"
 10 #import "HVWApp.h"
 11 #import "HVWDownloadImageOperation.h"
 12
 13 @interface ViewController () <UITableViewDataSource, UITableViewDelegate, HVWDownloadImageOperationDelegate>
 14
 15 /** 所有app数据 */
 16 @property(nonatomic, strong) NSArray *apps;
 17
 18 /** 任务队列 */
 19 @property(nonatomic, strong) NSOperationQueue *queue;
 20
 21 /** 所有任务 */
 22 @property(nonatomic, strong) NSMutableDictionary *operations;
 23
 24 /** 所有图片 */
 25 @property(nonatomic, strong) NSMutableDictionary *images;
 26
 27 @end
 28
 29 @implementation ViewController
 30
 31 /** 加载plist文件,读取数据到模型,存储到数组中 */
 32 - (NSArray *)apps {
 33     if (nil == _apps) {
 34         NSArray *dictArray = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"apps.plist" ofType:nil]];
 35
 36         NSMutableArray *appArray = [NSMutableArray array];
 37         for (NSDictionary *dict in dictArray) {
 38             HVWApp *app = [HVWApp appWithDictionary:dict];
 39             [appArray addObject:app];
 40         }
 41         _apps = appArray;
 42     }
 43     return _apps;
 44 }
 45
 46 - (NSOperationQueue *)queue {
 47     if (_queue == nil ) {
 48         _queue = [[NSOperationQueue alloc] init];
 49         _queue.maxConcurrentOperationCount = 3;
 50     }
 51     return _queue;
 52 }
 53
 54 - (NSMutableDictionary *)operations {
 55     if (nil == _operations) {
 56         _operations = [NSMutableDictionary dictionary];
 57     }
 58     return _operations;
 59 }
 60
 61 - (NSMutableDictionary *)images {
 62     if (nil == _images) {
 63         _images = [NSMutableDictionary dictionary];
 64     }
 65     return _images;
 66 }
 67
 68 - (void)viewDidLoad {
 69     [super viewDidLoad];
 70     // Do any additional setup after loading the view, typically from a nib.
 71 }
 72
 73 - (void)didReceiveMemoryWarning {
 74     [super didReceiveMemoryWarning];
 75     // Dispose of any resources that can be recreated.
 76 }
 77
 78 #pragma mark - tableViewDatasource
 79 - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
 80     return 1;
 81 }
 82
 83 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
 84     return self.apps.count;
 85 }
 86
 87 - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
 88     static NSString *ID = @"AppCell";
 89     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
 90
 91     if (nil == cell) {
 92         cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:ID];
 93     }
 94
 95     HVWApp *app = self.apps[indexPath.row];
 96     cell.textLabel.text = app.name;
 97     cell.detailTextLabel.text = app.download;
 98
 99     // 占位图片
100     cell.imageView.image = [UIImage imageNamed:@"a9ec8a13632762d0092abc3ca2ec08fa513dc619"];
101
102     // 如果没有图片,准备开启线程下载图片
103     UIImage *image = self.images[app.icon];
104
105     if (image) {
106         // 如果图片存在,不需要重复下载,直接设置图片
107         cell.imageView.image = image;
108     } else {
109
110         // 如果图片不存在,看看是不是正在下载
111         HVWDownloadImageOperation *operation = self.operations[app.icon];
112
113         if (operation) {
114             // 如果图片正在下载,不必要开启线的线程再进行下载
115         } else {
116
117             // 没有在下载,创建一个新的任务进行下载
118             operation = [[HVWDownloadImageOperation alloc] init];
119             // 设置代理
120             operation.delegate = self;
121             // 传送url
122             operation.url = app.icon;
123             // 记录indexPath
124             operation.indexPath = indexPath;
125
126             [self.queue addOperation:operation];
127
128             // 记录正在下载的任务
129             [self.operations setObject:operation forKey:operation.url];
130         }
131     }
132
133     return cell;
134 }
135
136 #pragma mark - HVWDownloadImageOperationDelegate
137 /** 代理方法,图片下载完成后,显示到cell上 */
138 - (void)downloadImageOperation:(HVWDownloadImageOperation *)operation didFinishedDownloadWithImage:(UIImage *)image {
139     UITableViewCell *cell = [self.tableView cellForRowAtIndexPath:operation.indexPath];
140     cell.imageView.image = image;
141     [self.tableView reloadRowsAtIndexPaths:@[operation.indexPath] withRowAnimation:UITableViewRowAnimationNone];
142
143     // 存储图片到内存
144     if (image) {
145         [self.images setObject:image forKey:operation.url];
146     }
147
148     NSLog(@"已经下载的图片数==========>%d", self.images.count);
149 }
150
151 @end
时间: 2024-10-10 07:32:49

[iOS 多线程 & 网络 - 1.3] - NSOperation的相关文章

[iOS 多线程 &amp; 网络 - 1.0] - 多线程概述

A.进程 什么是进程进程是指在系统中正在运行的一个应用程序 每个进程之间是独立的,每个进程均运行在其专用且受保护的内存空间内 比如同时打开QQ.Xcode,系统就会分别启动2个进程 通过"活动监视器"可以查看Mac系统中所开启的进程 B.线程 主线程.子线程占用内存分别是1M和512K 1.什么是线程1个进程要想执行任务,必须得有线程(每1个进程至少要有1条线程) 线程是进程的基本执行单元,一个进程(程序)的所有任务都在线程中执行 比如使用酷狗播放音乐.使用迅雷下载电影,都需要在线程中

[iOS 多线程 &amp; 网络 - 2.0] - 发送接收 服务器信息

A.搭建java服务器 使用eclipse.tomcat和struts2框架搭建一个简单的服务器 1.准备好合适版本的JDK.eclipse EE.tomcat.struts2 框架包 2.配置JDK和tomcat系统变量 3.在eclipse中创建一个Dynamic Web Project, 勾选创建web.xml 4.解压一个struts2中的app范例,参考其中的web.xml和struts.xml配置 5.配置tomcat,注意配置正确的服务器的路径和发布路径,不要使用默认的eclips

iOS多线程入门之NSThread,NSOperation,GCD

一 线程的概念 一个运行着的程序就是一个进程或者叫做一个任务,一个进程至少包含一个线程,线程就是程序的执行流.Mac和iOS中的程序启动,创建好一个进程的同时, 一个线程便开始运行,这个线程叫主线程.主线程在程序中的地位和其他线程不同,它是其他线程最终的父线程,且所有界面的显示操作即AppKit或 UIKit的操作必须在主线程进行. 系统中的每一个进程都有自己独立的虚拟内存空间,而同一个进程中的多个线程则共用进程的内存空间.每创建一个新的线程,都需要一些内存(如每个线程有自己的Stack空间)和

[iOS 多线程 &amp; 网络 - 2.9] - ASI框架

A.ASI基本知识 1.ASI简单介绍 ASI:全称是ASIHTTPRequest,外号“HTTP终结者”,功能十分强大. ASI的实现基于底层的CFNetwork框架,因此运行效率很高. ASI的github地址 https://github.com/pokeb/asi-http-request ASI的使用参考http://www.cnblogs.com/dotey/archive/2011/05/10/2041966.html http://www.oschina.net/question

[iOS 多线程 &amp; 网络 - 2.1] - 解析json

A.iOS中json的基本使用 1.解析json数据 (1)json反序列化 对象{}格式 {key : value, key : value,...} 的键值对的结构可以反序列化为OC中的NSDictionary数组[]格式 ["java","javascript","vb",...]可以反序列化为OC中的NSArray 提示JSON的数据格式与OC中的快速包装方法非常类似JSON的数据格式同样支持嵌套 (2)json工具 从iOS 5开始,使

[iOS 多线程 &amp; 网络 - 2.8] - 检测网络状态

A.说明 在网络应用中,需要对用户设备的网络状态进行实时监控,有两个目的:(1)让用户了解自己的网络状态,防止一些误会(比如怪应用无能)(2)根据用户的网络状态进行智能处理,节省用户流量,提高用户体验 WIFI\3G网络:自动下载高清图片 低速网络:只下载缩略图 没有网络:只显示离线的缓存数据 苹果官方提供了一个叫Reachability的示例程序,便于开发者检测网络状态https://developer.apple.com/library/ios/samplecode/Reachability

iOS 多线程(三)NSOperation

NSOperation NSOperation是苹果封装的一套多线程的东西,不像GCD是纯C语言的,这个是OC的.但相比较之下GCD会更快一些,但本质上NSOPeration是多GDC的封装. NSOperation相对于GCD: NSOperation拥有更多的函数可用 NSOperationQueue中,可以建立各个NSOperation之间的依赖关系. NSOperationQueue支持KVO.可以监测operation是否正在执行(isExecuted).是否结束(isFinished

IOS多线程(NSThread,NSOperation,Grand Central Dispatch)

•NSThread: –优点:NSThread 比其他两个轻量级,使用简单 –缺点:需要自己管理线程的生命周期.线程同步.加锁.睡眠以及唤醒等.线程同步对数据的加锁会有一定的系统开销 •NSOperation: –不需要关心线程管理,数据同步的事情,可以把精力放在自己需要执行的操作上 –NSOperation是面向对象的 - (void)performSelectorOnMainThread:(SEL)aSelector withObject:(id)arg waitUntilDone:(BOO

iOS 多线程(4)NSOperation

2.NSOperation 2.1 NSOperation基本使用 (1)相关概念 01 NSOperation是对GCD的包装 02 两个核心概念[队列+操作] (2)基本使用 01 NSOperation本身是抽象类,只能只有它的子类 02 三个子类分别是:NSBlockOperation.NSInvocationOperation以及自定义继承自NSOperation的类 03 NSOperation和NSOperationQueue结合使用实现多线程并发 (3)相关代码 // 01 NS