多线程NSOperation--自定义非并行的 NSOperation(二)

22 篇文章 0 订阅

转自:http://www.jianshu.com/p/813f7d58935d


1 自定义非并行的 NSOperation

前文介绍过 NSInvocationOperation 和 NSBlockOperation 都继承自NSOperation类。

我们亦可以通过继承 NSOperation 类,来自定义非并行的 Operation。

@interface VinnyOperation : NSOperation
@end

头文件很简单,只需要继承 NSOperation ,可根据实际需要决定是否需要自定义init方法。而且仅仅需要自定义main方法,将需要执行的操作写在main方法中。

#import "VinnyOperation.h"

@implementation VinnyOperation
- (void)main
{
    NSLog(@"main begin");
    @try {
        // 提供一个变量标识,来表示需要执行的操作是否完成了,当然,没开始执行之前,为NO
        BOOL taskIsFinished = NO;
        // while 保证:只有当没有执行完成和没有被取消,才执行自定义的相应操作
        while (taskIsFinished == NO && [self isCancelled] == NO){
            // 自定义的操作
            sleep(10);  // 睡眠模拟耗时操作
            NSLog(@"currentThread = %@", [NSThread currentThread]);
            NSLog(@"mainThread    = %@", [NSThread mainThread]);
            // 这里相应的操作都已经完成,后面就是要通知KVO我们的操作完成了。
            taskIsFinished = YES;
        }
    }
    @catch (NSException * e) {
        NSLog(@"Exception %@", e);
    }
    NSLog(@"main end");
}
@end

使用的时候也非常简单

    VinnyOperation *op = [[VinnyOperation alloc] init];
    NSLog(@"start before");
    [op start];
    NSLog(@"start after");

看一下控制台打印的结果

2015-12-29 19:21:56.895 test[67010:50712745] start before
2015-12-29 19:21:56.896 test[67010:50712745] main begin
2015-12-29 19:22:06.900 test[67010:50712745] currentThread = <NSThread: 0x7fc560d044d0>{number = 1, name = main}
2015-12-29 19:22:06.900 test[67010:50712745] mainThread    = <NSThread: 0x7fc560d044d0>{number = 1, name = main}
2015-12-29 19:22:06.900 test[67010:50712745] main end
2015-12-29 19:22:06.900 test[67010:50712745] start after

可以看出是main方法是非并行的,而且执行的操作与调用start是在同一个线程中。

2 自定义并行的 NSOperation

自定义并行的 NSOperation 则要复杂一点,首先必须重写以下几个方法:

  • start: 所有并行的 Operations 都必须重写这个方法,然后在你想要执行的线程中手动调用这个方法。注意:任何时候都不能调用父类的start方法。
  • main: 在start方法中调用,但是注意要定义独立的自动释放池与别的线程区分开。
  • isExecuting: 是否执行中,需要实现KVO通知机制。
  • isFinished: 是否已完成,需要实现KVO通知机制。
  • isConcurrent: 该方法现在已经由isAsynchronous方法代替,并且 NSOperationQueue 也已经忽略这个方法的值。
  • isAsynchronous: 该方法默认返回 NO ,表示非并发执行。并发执行需要自定义并且返回 YES。后面会根据这个返回值来决定是否并发。

与非并发操作不同的是,需要另外自定义一个方法来执行操作而不是直接调用start方法

@interface MyOperation : NSOperation
- (BOOL)performOperation:(NSOperation*)anOp;    // 执行操作调用这个方法
@end

实现其中的必要方法:

#import "MyOperation.h"

@interface MyOperation () {
    BOOL        executing;  // 执行中
    BOOL        finished;   // 已完成
}
@end

@implementation MyOperation
- (id)init {
    self = [super init];
    if (self) {
        executing = NO;
        finished = NO;
    }
    return self;
}

- (void)start {
    // Always check for cancellation before launching the task.
    if ([self isCancelled])
    {
        // Must move the operation to the finished state if it is canceled.
        [self willChangeValueForKey:@"isFinished"];
        finished = YES;
        [self didChangeValueForKey:@"isFinished"];
        return;
    }

    // If the operation is not canceled, begin executing the task.
    [self willChangeValueForKey:@"isExecuting"];
    [NSThread detachNewThreadSelector:@selector(main) toTarget:self withObject:nil];
    executing = YES;
    [self didChangeValueForKey:@"isExecuting"];
}

- (void)main {
    NSLog(@"main begin");
    @try {
        // 必须为自定义的 operation 提供 autorelease pool,因为 operation 完成后需要销毁。
        @autoreleasepool {
            // 提供一个变量标识,来表示需要执行的操作是否完成了,当然,没开始执行之前,为NO
            BOOL taskIsFinished = NO;
            // while 保证:只有当没有执行完成和没有被取消,才执行自定义的相应操作
            while (taskIsFinished == NO && [self isCancelled] == NO){
                // 自定义的操作
                //sleep(10);  // 睡眠模拟耗时操作
                NSLog(@"currentThread = %@", [NSThread currentThread]);
                NSLog(@"mainThread    = %@", [NSThread mainThread]);

                // 这里相应的操作都已经完成,后面就是要通知KVO我们的操作完成了。
                taskIsFinished = YES;
            }
            [self completeOperation];

        }
    }
    @catch (NSException * e) {
        NSLog(@"Exception %@", e);
    }
    NSLog(@"main end");
}

- (void)completeOperation {
    [self willChangeValueForKey:@"isFinished"];
    [self willChangeValueForKey:@"isExecuting"];

    executing = NO;
    finished = YES;

    [self didChangeValueForKey:@"isExecuting"];
    [self didChangeValueForKey:@"isFinished"];
}

// 已经弃用,使用 isAsynchronous 代替
//- (BOOL)isConcurrent {
//    return NO;
//}

- (BOOL)isAsynchronous {
    return YES;
}

- (BOOL)isExecuting {
    return executing;
}

- (BOOL)isFinished {
    return finished;
}

// 执行操作
- (BOOL)performOperation:(NSOperation*)anOp
{
    BOOL        ranIt = NO;

    if ([anOp isReady] && ![anOp isCancelled])
    {
        if (![anOp isAsynchronous]) {
            [anOp start];
        }
        else {
            [NSThread detachNewThreadSelector:@selector(start)
                                     toTarget:anOp withObject:nil];
        }
        ranIt = YES;
    }
    else if ([anOp isCancelled])
    {
        // If it was canceled before it was started,
        //  move the operation to the finished state.
        [self willChangeValueForKey:@"isFinished"];
        [self willChangeValueForKey:@"isExecuting"];
        executing = NO;
        finished = YES;
        [self didChangeValueForKey:@"isExecuting"];
        [self didChangeValueForKey:@"isFinished"];

        // Set ranIt to YES to prevent the operation from
        // being passed to this method again in the future.
        ranIt = YES;
    }
    return ranIt;
}

使用这个 Operation 如下:

    MyOperation *op = [[MyOperation alloc] init];
    NSLog(@"start before");
    [op performOperation:op];
    NSLog(@"start after");

看一下控制台打印的结果

2015-12-29 20:01:53.130 test[27083:51105353] start before
2015-12-29 20:01:53.131 test[27083:51105353] start after
2015-12-29 20:01:53.131 test[27083:51105608] main begin
2015-12-29 20:01:53.131 test[27083:51105608] currentThread = <NSThread: 0x7ff148d976d0>{number = 3, name = (null)}
2015-12-29 20:01:53.131 test[27083:51105608] mainThread    = <NSThread: 0x7ff148e01250>{number = 1, name = (null)}
2015-12-29 20:01:53.131 test[27083:51105608] main end

可以看到这个操作是并发执行,并且是一个独立的线程。

3 总结

  1. 如果需要自定义并发执行的 Operation,必须重写 startmainisExecutingisFinishedisAsynchronous 方法。
  2. 在 operation 的 main 方法里面,必须提供 autorelease pool,因为你的 operation 完成后需要销毁。
  3. 一旦你的 operation 开始了,必须通过 KVO,告诉所有的监听者,现在该operation的执行状态。
  4. 调用时,如果需要并发执行 Operation,必须调用performOperation:方法,当然,也可以改为自定义其他方法或者直接在start方法添加多线程调用。
  5. 对于自定义的 Operation 类,如果不需要并发执行,可以直接调用start方法。

4 尾巴

刚开始看 NSOperation 的文档没搞明白怎么自定义多线程的操作,文档里只是说需要自定义 isExecutingisFinishedisConcurrentisAsynchronous 这四个方法,然后说根据 isAsynchronous 的返回值来判断是否多线程,我以为只要重写这个方法的时候返回 YES 就行了,NSOperation 就会自动多线程执行了,但是测试发现却不是这样的,多线程还得自己去创建再使用。

再有就是自定义多线程的 NSOperation 时,还必须自己管理其中表示状态的成员,而且需要实现 KVO 机制,使得这个过程复杂化了。

其实在大多数时候我们并不会直接去使用自定义的 NSOperation ,如果操作不复杂,可以直接使用 NSInvocationOperation 和 NSBlockOperation 这两个子类,那就直接用了,如果复杂一些,必须自定义又需要多线程,通常都会用 NSOperationQueue 来包装,使用起来更加简洁,下一篇会详细介绍 NSOperationQueue 的使用。



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。
在网络编程中,多线程编程是一种常用的技术,可以提高程序的并发性和性能。下面是一些关于多线程编程的常用方法和注意事项: 1. NSThread:NSThread是iOS中最底层的线程类,它可以通过类方法或实例方法来创建线程。使用NSThread可以设置线程的名称、优先级,以及控制线程的睡眠和退出等操作。 2. 线程调度:在多线程编程中,多个线程会并发运行,但线程的执行顺序是由CPU调度器决定的,程序员无法控制。多个线程会同时竞争CPU资源,谁先抢到资源谁就先执行,所以多线程的执行顺序是随机的。 3. 多线程的创建:在iOS开发中,常用的多线程编程方式有三种:NSThread、GCD和NSOperationNSThread是最底层的线程类,可以直接操作线程的各种属性和方法。GCD(Grand Central Dispatch)提供了一种高效的并发编程模型,可以通过队列来管理任务的执行。NSOperation是基于GCD的更高层次的封装,提供了更多的控制和管理线程的功能。 4. 线程的创建顺序:在多线程编程中,并不能保证哪个线程会先运行,即无法确定新创建的线程或调用线程哪个会先执行。新创建的线程可以访问进程的地址空间,并继承调用线程的浮点环境和信号屏蔽字,但挂起信号集会被清除。 总结来说,多线程编程是一种提高程序并发性和性能的技术,在网络编程中尤为重要。通过使用NSThread、GCD或NSOperation等方法,可以实现多线程的创建和管理。然而,程序员无法控制线程的执行顺序,因为线程的调度是由CPU调度器决定的。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [IOS之多线程基础(OC)](https://blog.csdn.net/yong_19930826/article/details/105857055)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [UNIX环境高级编程笔记](https://blog.csdn.net/w_x_myself/article/details/128613534)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值