iOS开发之多线程NSOperation

本文介绍iOS开发中利用NSOperation及NSOperationQueue进行多线程编程的方法,包括NSOperation的基本使用、NSInvocationOperation与NSBlockOperation的具体实现方式及自定义NSOperation的步骤。

       之前在介绍iOS开发中多线程实现方案GCD的时候,说到GCD是纯C语言的,且是底层API。我们用起来还是有点吃力,所有苹果推了NSOperation方案来实现多线程。之前提到,NSOperation是基于GCD的,所以GCD自动创建线程,自动管理线程生命周期等优点NSOperation都有,且比GCD多了一些功能。下面我们来学习NSOperation的相关知识。

NSOperation的作用

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

NSOperation和NSOperationQueue实现多线程的具体步骤

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

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

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

4.    将取出的NSOperation封装的操作放到一条新线程中执行

NSOperation的子类

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

使用NSOperation子类的方式有3种

NSInvocationOperation

NSBlockOperation

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

NSInvocationOperation

创建NSInvocationOperation对象

- (id)initWithTarget:(id)targetselector:(SEL)sel object:(id)arg;

调用start方法开始执行操作

- (void)start;

一旦执行操作,就会调用target的sel方法

注意

默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作

只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作

例子:(直接调用start方法执行操作
- (void)testInvocation {
    // 创建操作
    NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage) object:nil];
    
    // 执行操作(默认在当前线程执行)
    [operation start];
}

- (void)downLoadImage {
    NSLog(@"Invocation-downLoadImage-%@", [NSThread currentThread]);
}

运行结果:

例子:(将操作添加到NSOperationQueue

- (void)testInvocation {
    // 创建操作
    NSInvocationOperation *operation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage1) object:nil];
    NSInvocationOperation *operation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage2) object:nil];
    NSInvocationOperation *operation3 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage3) object:nil];
    
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];

    // 将操作添加到队列
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
}

- (void)downLoadImage1 {
    NSLog(@"Invocation-downLoadImage1-%@", [NSThread currentThread]);
}
- (void)downLoadImage2 {
    NSLog(@"Invocation-downLoadImage2-%@", [NSThread currentThread]);
}
- (void)downLoadImage3 {
    NSLog(@"Invocation-downLoadImage3-%@", [NSThread currentThread]);
}
运行结果:

NSBlockOperation

创建NSBlockOperation对象

+ (id)blockOperationWithBlock:(void(^)(void))block;

 

通过addExecutionBlock:方法添加更多的操作

- (void)addExecutionBlock:(void(^)(void))block;

 

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

例子:(操作数 == 1,默认在当前线程中执行

- (void)testBlock {
    // 封装操作
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        // 操作
        NSLog(@"block-downLoad-%@", [NSThread currentThread]);
    }];
    
    [operation start];
}
运行结果:

例子:(操作数大于1,开启新的线程执行

- (void)testBlock {

   // 封装操作

   NSBlockOperation *operation = [NSBlockOperationblockOperationWithBlock:^{

       // 操作

       NSLog(@"block-downLoad1-%@", [NSThread currentThread]);

   }];

   // 添加操作
   [operation addExecutionBlock:^{

       NSLog(@"block-downLoad2-%@",[NSThread currentThread]);

   }];

   [operation addExecutionBlock:^{

       NSLog(@"block-downLoad3-%@", [NSThread currentThread]);

   }];

   [operation start];

}
运行结果:

NSOperationQueue

NSOperationQueue的作用

NSOperation可以调用start方法来执行任务,但默认是同步执行的

如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动步执行NSOperation中的操作

 

添加操作到NSOperationQueue中

-(void)addOperation:(NSOperation *)op;// 添加封装后的操作

- (void)addOperationWithBlock:(void(^)(void))block;// 将确定操作添加到队列中

例子:(向队列中添加NSInvocationOperationNSBlockOperation操作

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.queue = [[NSOperationQueue alloc] init];
    
    [self testBlock];
    [self testInvocation];
    
    
}
- (void)testBlock {
    // 封装操作
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        // 操作
        NSLog(@"block-downLoad1-%@", [NSThread currentThread]);
    }];
    
    // 添加操作
    [operation addExecutionBlock:^{
        NSLog(@"block-downLoad2-%@", [NSThread currentThread]);
    }];
    [operation addExecutionBlock:^{
        NSLog(@"block-downLoad3-%@", [NSThread currentThread]);
    }];
    
    // 将操作添加到队列
    [self.queue addOperation:operation];
}

- (void)testInvocation {
    // 创建操作
    NSInvocationOperation *operation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage1) object:nil];
    NSInvocationOperation *operation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage2) object:nil];
    NSInvocationOperation *operation3 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(downLoadImage3) object:nil];
    
    // 将操作添加到队列
    [self.queue addOperation:operation1];
    [self.queue addOperation:operation2];
[self.queue addOperation:operation3];
//    [self.queue addOperations:@[operation1, operation2, operation3] waitUntilFinished:NO];// YES:等3个操作全部执行完,这行代码才会过;NO:
}

- (void)downLoadImage1 {
    NSLog(@"Invocation-downLoadImage1-%@", [NSThread currentThread]);
}
- (void)downLoadImage2 {
    NSLog(@"Invocation-downLoadImage2-%@", [NSThread currentThread]);
}
- (void)downLoadImage3 {
    NSLog(@"Invocation-downLoadImage3-%@", [NSThread currentThread]);
}

注意:[self.queueaddOperations:@[operation1, operation2, operation3] waitUntilFinished:NO];//YES:3个操作全部执行完,该行代码才算执行完;NO:只要操作添加完该行代码就执行完。

运行结果:


最大并发数

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

最大并发数的相关方法

- (NSInteger)maxConcurrentOperationCount;

- (void)setMaxConcurrentOperationCount:(NSInteger)cnt;

队列的取消、暂停、恢复

取消队列的所有操作

- (void)cancelAllOperations;

提示:也可以调用NSOperation- (void)cancel方法取消单个操作

 

暂停和恢复队列

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

- (BOOL)isSuspended;

操作优先级

设置NSOperation在queue中的优先级,可以改变操作的执行优先级

- (NSOperationQueuePriority)queuePriority;

-(void)setQueuePriority:(NSOperationQueuePriority)p;
// 优先级的取值
// NSOperationQueuePriorityVeryLow = -8L,
// NSOperationQueuePriorityLow = -4L,
// NSOperationQueuePriorityNormal = 0,
// NSOperationQueuePriorityHigh = 4,
// NSOperationQueuePriorityVeryHigh = 8


操作依赖

NSOperation之间可以设置依赖来保证执行顺序

比如一定要让操作A执行完后,才能执行操作B,可以这么写

[operationB addDependency:operationA]; // 操作B依赖于操作A

注意:不能相互依赖,比如A依赖BB依赖A

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


操作的监听

可以监听一个操作的执行完毕

- (void (^)(void))completionBlock;

- (void)setCompletionBlock:(void(^)(void))block;

自定义NSOperation

自定义NSOperation的步骤很简单

 重写- (void)main方法,在里面实现想执行的任务

 

 重写- (void)main方法的注意点

 自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)

 经常通过- (BOOL)isCancelled方法检测操作是否被取消,对取消做出响应

NSOperation是苹果官方推荐使用的多线程实现方案,所以要尽量掌握这种方案。




内容概要:本文系统介绍了算术优化算法(AOA)的基本原理、核心思想及Python实现方法,并通过图像分割的实际案例展示了其应用价值。AOA是一种基于种群的元启发式算法,其核心思想来源于四则运算,利用乘除运算进行全局勘探,加减运算进行局部开发,通过数学优化器加速函数(MOA)和数学优化概率(MOP)动态控制搜索过程,在全局探索与局部开发之间实现平衡。文章详细解析了算法的初始化、勘探与开发阶段的更新策略,并提供了完整的Python代码实现,结合Rastrigin函数进行测试验证。进一步地,以Flask框架搭建前后端分离系统,将AOA应用于图像分割任务,展示了其在实际工程中的可行性与高效性。最后,通过收敛速度、寻优精度等指标评估算法性能,并提出自适应参数调整、模型优化和并行计算等改进策略。; 适合人群:具备一定Python编程基础和优化算法基础知识的高校学生、科研人员及工程技术人员,尤其适合从事人工智能、图像处理、智能优化等领域的从业者;; 使用场景及目标:①理解元启发式算法的设计思想与实现机制;②掌握AOA在函数优化、图像分割等实际问题中的建模与求解方法;③学习如何将优化算法集成到Web系统中实现工程化应用;④为算法性能评估与改进提供实践参考; 阅读建议:建议读者结合代码逐行调试,深入理解算法流程中MOA与MOP的作用机制,尝试在不同测试函数上运行算法以观察性能差异,并可进一步扩展图像分割模块,引入更复杂的预处理或后处理技术以提升分割效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值