NSOperation

2016-04-28  本文已影响411人  月下独酌灬

NSOperation

NSOperation简介

GCD和OP的关系图.png

NSOperation类的介绍

  1. NSOperation是个抽象类,无法直接使用.因为方法只有声明没有实现.

    作为父类使用的.约束子类共有的属性和方法.

  2. 子类 :

    • NSInvocationOperation
    • NSBlockOperation
    • 自定义NSOperation

    操作默认是异步的.

  3. 队列 : NSOperationQueue

    队列默认是并发的.

  4. 核心 :

    • GCD的核心 : 将任务添加到队列
    • OP的核心 : 将操作添加到队列

使用步骤

  1. 先将需要执行的操作封装到一个NSOperation对象中.创建NSOperation对象.
  2. 将NSOperation对象添加到NSOperationQueue中.
  3. NSOperationQueue会自动将NSOperation取出来.
  4. 将取出的NSOperation封装的操作自动放到一条对应的新线程中执行.
操作添加到队列.png

NSInvocationOperation

NSInvocationOperation 基本使用演练

- (void)demo:(id)parram
{
    // 查看当前线程
    NSLog(@"%@ %@",parram,[NSThread currentThread]);
}

OP调用start方法

- (void)opDemo1
{
    // 创建操作对象
    NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@"InvocationOperation"];
    // 调用start方法
    [op start];
}

NSInvocationOperation 将操作添加到队列

- (void)opDemo2
{
    // 操作对象 : OP中的操作对象默认是异步执行
    NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@"InvocationOperation"];
    // 队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    // 将操作添加到队列
    [queue addOperation:op];
}

NSInvocationOperation 验证队列并发性

- (void)opDemo3
{
    // 队列 : 默认是并发的
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];

    // 循环的向队列中添加10个操作
    for (int i = 0; i < 10; i++) {
        // 操作对象 : OP中的操作对象默认是异步执行
        NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@(i)];
        // 将操作添加到队列
        [queue addOperation:op];
    }
}

NSBlockOperation

NSBlockOperation 基本使用演练

NSBlockOperation 操作添加到队列

- (void)opDemo1
{
    // 队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    // 操作 : 默认是异步的
    NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
        // 查看当前线程
        NSLog(@"%@",[NSThread currentThread]);
    }];
    // 将操作添加到队列
    [queue addOperation:op];

    // 在当前线程执行
//    [op start];
}

NSBlockOperation 验证队列并发性

- (void)opDemo2
{
    // 队列 : 默认是并发
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];

    for (int i = 0; i < 10; i++) {
        // 操作 : 默认是异步的
        NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
            // 查看当前线程
            NSLog(@"%d %@",i,[NSThread currentThread]);
        }];
        // 将操作添加到队列
        [queue addOperation:op];
    }
}

开发建议

/// 定义全局队列
@property (nonatomic,strong) NSOperationQueue *queue;
- (NSOperationQueue *)queue
{
    if (_queue==nil) {
        _queue = [[NSOperationQueue alloc] init];
    }
    return _queue;
}

NSBlockOperation 简写

- (void)opDemo3
{
    [self.queue addOperationWithBlock:^{
        // 查看当前线程
        NSLog(@"%@",[NSThread currentThread]);
    }];
}

线程间通信

- (void)opDemo5
{
    [self.queue addOperationWithBlock:^{
        NSLog(@"努力下载中...%@",[NSThread currentThread]);

        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            NSLog(@"更新UI...%@",[NSThread currentThread]);
        }];
    }];
}

NSOperation与GCD对比

GCD

NSOperation

NSOperation高级功能演练

队列的最大并发数

*** 准备队列**

/// 定义全局队列
@property (nonatomic,strong) NSOperationQueue *queue;
- (NSOperationQueue *)queue
{
    if (_queue==nil) {
        _queue = [[NSOperationQueue alloc] init];

        // 设置最大并发数 : 每次只能调度两个操作执行
        _queue.maxConcurrentOperationCount = 2;
    }
    return _queue;
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self opDemo];
}
- (void)opDemo
{
    NSLog(@"start");

    for (int i = 0; i < 20; i++) {
        [self.queue addOperationWithBlock:^{

            // 休眠一秒钟,演示最大并发数的效果更好
            [NSThread sleepForTimeInterval:1.0];

            NSLog(@"%d %@",i,[NSThread currentThread]);
        }];
    }
}

执行的结果 : 任务是两个两个的执行.

队列的暂停继续和取消全部

相关的属性和方法介绍

1. 队列暂停

#pragma mark - 演示队列的暂停
- (IBAction)stop:(id)sender
{
    // 暂停队列之前判断队列中有无操作
    if (self.queue.operationCount == 0) {
        return;
    }

    // 暂停队列
    self.queue.suspended = YES;
    NSLog(@"暂停 %zd",self.queue.operationCount);
}

注意 : 如果先暂停队列,再添加操作到队列,队列不会调度添加的操作.所以在暂停队列之前要判断队列中有没有任务.如果没有任务就不暂停队列.

2. 队列继续

- (IBAction)resume:(id)sender
{
    // 队列继续
    self.queue.suspended = NO;
    NSLog(@"继续 %zd",self.queue.operationCount);
}

3. 队列取消全部

- (IBAction)cancelAll:(id)sender
{
    [self.queue cancelAllOperations];
    NSLog(@"取消全部 %zd",self.queue.operationCount);
}

操作优先级和监听操作完成回调

- (void)opDemo
{
    // 操作1
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        for (int i = 0; i < 10; i++) {
            // 查看当前线程
            NSLog(@"op1 %@",[NSThread currentThread]);
        }
    }];

    //设置操作的优先级
    op1.qualityOfService = NSQualityOfServiceUserInteractive;

    // 当操作执行结束之后,就会回调,是在子线程中执行的
    [op1 setCompletionBlock:^{
        // 查看当前线程
        NSLog(@"操作结束了 %@",[NSThread currentThread]);
    }];

    [self.queue addOperation:op1];

    // 操作2
    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        for (int i = 0; i < 10; i++) {
            // 查看当前线程
            NSLog(@"op2 %@",[NSThread currentThread]);
        }
    }];
    op2.qualityOfService = NSQualityOfServiceBackground;
    [self.queue addOperation:op2];
}

操作间依赖

操作op2依赖于op1
[op2 addDependency:op1];

4. 操作依赖

需求 : 登陆-->付费-->下载-->通知用户

#pragma mark - 操作依赖
- (void)dependency
{
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"登陆 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"付费 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"下载 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op4 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"通知用户 %@",[NSThread currentThread]);
    }];
}
// 操作2依赖操作1
[op2 addDependency:op1];
[op3 addDependency:op2];
[op4 addDependency:op3];

// 不能循环依赖 : 操作不会被调度
// [op1 addDependency:op4];

// waitUntilFinished : 是否等到指定的操作执行结束再执行后面的代码
[self.queue addOperations:@[op1,op2,op3,op4] waitUntilFinished:NO];

// 验证 waitUntilFinished
NSLog(@"end");
// 操作2依赖操作1
[op2 addDependency:op1];
[op3 addDependency:op2];
[op4 addDependency:op3];

// 不能循环依赖 : 操作不会被调度
// [op1 addDependency:op4];

// waitUntilFinished : 是否等到指定的操作执行结束再执行后面的代码
[self.queue addOperations:@[op1,op2,op3] waitUntilFinished:NO];

// 通知用户的操作在主线程中执行
// 操作可以跨队列依赖
[[NSOperationQueue mainQueue] addOperation:op4];

// 验证 waitUntilFinished
NSLog(@"end");
上一篇 下一篇

猜你喜欢

热点阅读