「OC」多线程(三)——NSOperation

news2025/1/17 17:58:56

「OC」多线程(三)——NSOperation

文章目录

  • 「OC」多线程(三)——NSOperation
    • 前言
    • 介绍
      • 实现的具体步骤
    • NSOperation的创建
    • NSOperationQueue的使用
    • 使用实例
      • NSInvocationOperation的使用
      • NSBlockOperation的使用
      • NSOperationQueue的使用
      • 取消操作
      • 最大并发数
    • 自定义NSOperation子类
      • 相关属性和方法
      • start方法
      • main方法
      • 重写NSOperation实现并发
    • 参考文章

前言

在写了知乎日报之前学习了相关的GCD和NSThread的相关内容,然后在知乎日报写完之后,我继续开始OC之中多线程的NSOperation的学习。由于之后肯定是要对AFNetworkingSDWebImage的源码继续相关学习,所以学习NSOperation的相关内容也是给后面做铺垫。

介绍

NSOperation 是苹果公司对 GCD 的封装,完全面向对象,并比GCD多了一些更简单实用的功能,所以使用起来更加方便易于理解。NSOperation 和NSOperationQueue 分别对应 GCD 的 任务 和 队列。

首先我们来看看,关于GPT给出的GCD和NSOperation的区别

特性NSOperationGCD
抽象级别面向对象,高抽象基于 C 语言,低抽象
任务依赖支持,任务之间可设置依赖关系不支持,需要手动控制
取消任务支持任务取消并可检测取消状态不支持(任务一旦开始,无法取消)
任务管理可管理任务状态(开始、取消、完成等)无任务状态管理
队列控制最大并发数、优先级、依赖性控制较少,需手动实现

实现的具体步骤

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

  2. 将NSOperation对象添加到NSOperationQueue中系统会自动将NSOperationQueue中的NSOperation取出来,并将取出的NSOperation封装的操作放到一条新线程中执行

NSOperation的创建

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

  1. NSInvocationOperation
    /*
     第一个参数:目标对象
     第二个参数:选择器,要调用的方法
     第三个参数:方法要传递的参数
     */
NSInvocationOperation *op  = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download) object:nil];
//启动操作
[op start];
  1. NSBlockOperation(最常用)
//1.封装操作
NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
       //要执行的操作,在主线程中执行
       NSLog(@"1------%@",[NSThread currentThread]); 
}];
//2.追加操作,追加的操作在子线程中执行,可以追加多条操作
[op addExecutionBlock:^{
        NSLog(@"---download2--%@",[NSThread currentThread]);
    }];
[op start];
  1. 自定义子类继承NSOperation,实现内部相应的方法
// 重写自定义类的main方法实现封装操作
-(void)main
{
    // 要执行的操作
}
// 实例化一个自定义对象,并执行操作
JCOperation *op = [[JCOperation alloc]init];
[op start];

自定义类封装性高,复用性高。

NSOperationQueue的使用

NSOperation中的两种队列

  • 主队列:通过mainQueue获得,凡是放到主队列中的任务都将在主线程执行
  • 非主队列:直接alloc init出来的队列。非主队列同时具备了并发和串行的功能,通过设置最大并发数属性来控制任务是并发执行还是串行执行
// 主队列获取方法
NSOperationQueue *queue = [NSOperationQueue mainQueue];
// 自定义队列创建方法
NSOperationQueue *queue = [[NSOperationQueue alloc] init];

NSOperationQueue的作用

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

添加操作到NSOperationQueue中

- (void)addOperation:(NSOperation *)op;
- (void)addOperationWithBlock:(void (^)(void))block;

注意:将操作添加到NSOperationQueue中,就会自动启动,不需要再自己启动了,addOperation 内部调用 start方法 ,而start方法内部调用 main方法

使用实例

了解了以上相关的原理,接下来学习几个例子进行巩固

NSInvocationOperation的使用

NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(taskMethod) object:nil];
[operation start]; // 直接启动任务

- (void)taskMethod {
    NSLog(@"任务执行中 - %@", [NSThread currentThread]);
}

这样的操作,默认任务是要在当前主线程之中完成的,如果我们想要实现异步操作的话,就需要和NSOperationQueue配合进行使用,不过这样一来代码就会较为的冗余,因此在实际运用之中我们不太使用。

NSBlockOperation的使用

NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"任务1 - %@", [NSThread currentThread]);
}];

// 添加额外任务
[operation addExecutionBlock:^{
    NSLog(@"任务2 - %@", [NSThread currentThread]);
}];

[operation start];

任务的主块和附加块可能并发执行。而且,主任务是在主线程之中执行,而额外的任务则是在子线程之中执行

image-20241125201345447

NSOperationQueue的使用

NSOperationQueue *queue = [[NSOperationQueue alloc] init];

// 创建任务
NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"任务1 - %@", [NSThread currentThread]);
}];
NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"任务2 - %@", [NSThread currentThread]);
}];
NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"任务3 - %@", [NSThread currentThread]);
}];

// 将任务加入队列
[queue addOperation:operation1];
[queue addOperation:operation2];
[queue addOperation:operation3];


  • 队列会自动创建多个线程并发执行任务。

  • 默认并发执行任务,可以通过设置maxConcurrentOperationCount调整并发数。

image-20241125201658428

取消操作

我们在任务进行的时候通过调用cancel的方法对任务进行取消,不过值得注意的是,当我们取消任务的时候,任务并不会自动停止,而是需要我们通过调用任务的isCancelled的方法进行判断,我们用一个简单的例子实现方式如下

NSOperationQueue *queue = [[NSOperationQueue alloc] init];
__block NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
    for (int i = 0; i < 5; i++) {
        if ([operation isCancelled]) {
            NSLog(@"任务被取消");
            return;
        }
        NSLog(@"任务执行中:%d", i);
        [NSThread sleepForTimeInterval:1]; // 模拟耗时任务
    }
    NSLog(@"任务完成");
}];

[queue addOperation:operation];

// 2秒后取消任务
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    [operation cancel];
});

image-20241125205311226

我们可以将if ([operation isCancelled]) 的内容注释,这时候我们可以发现,任务会直接执行完整,而不是到任务2的时候就中断了。

最大并发数

前面我们说到了,NSOperationQueue之中每添加一个任务,都会将任务添加到一个子线程之中,那如果不想要让这个队列开这么多个并发的线程,我们就可以设置这个队列之中的属性maxConcurrentOperationCount

NSOperationQueue *queue = [[NSOperationQueue alloc] init];
queue.maxConcurrentOperationCount = 2; // 最多并发2个任务

for (int i = 0; i < 5; i++) {
    [queue addOperationWithBlock:^{
        NSLog(@"任务%d - %@", i, [NSThread currentThread]);
        [NSThread sleepForTimeInterval:1]; // 模拟耗时任务
    }];
}

image-20241125214528178

自定义NSOperation子类

当内置的 NSBlockOperationNSInvocationOperation 无法满足需求时,可以继承 NSOperation 自定义复杂任务。

相关属性和方法

// 对于并发的Operation需要重写改方法
- (void)start;

// 非并发的Operation需要重写该方法
- (void)main;

// 任务是否取消(只读) 自定义子类,需重写该属性
@property (readonly, getter=isCancelled) BOOL cancelled;

// 可取消操作,实质是标记 isCancelled 状态,自定义子类,需利用该方法标记取消状态
- (void)cancel;

// 任务是否正在执行(只读),自定义子类,需重写该属性
@property (readonly, getter=isExecuting) BOOL executing;

// 任务是否结束(只读),自定义子类,需重写该属性
// 如果为YES,则队列会将任务移除队列
@property (readonly, getter=isFinished) BOOL finished;

// 判断任务是否为并发(只读),默认返回NO
// 自定义子类,需重写getter方法,并返回YES
@property (readonly, getter=isAsynchronous) BOOL asynchronous;

// 任务是否准备就绪(只读)
// 对于加入队列的任务来说,ready为YES,则表示该任务即将开始执行
// 如果存在依赖关系的任务没有执行完,则ready为NO
@property (readonly, getter=isReady) BOOL ready;

对于NSOperation.如果要自定义的话,我们需要重写start或者main,这两个方法的其中之一,自定义的是并发的NSOperation.那么我们得重写start方法.如果是非并发的话就只需要将相关逻辑写在main方法之中。接下来我们介绍一下相关的main和start方法的区别

start方法

start 方法是 NSOperation 的入口点,表示任务开始执行。

当调用 start 方法时,NSOperation 会根据其内部状态决定如何执行任务:

  • 如果操作已被取消(即 isCancelled == YES),它会直接标记为完成,而不会执行。
  • 如果未被取消:
    • 在主线程中调用 main 方法(默认情况下)。
    • 如果操作是异步的(需要自定义实现),start 方法通常会启动一个异步任务

默认的start方法逻辑大致如下

- (void)start {
    if (self.isCancelled) {
        [self finish];
        return;
    }

    [self setIsExecuting:YES];
    [self main];
    [self setIsExecuting:NO];
    [self finish];
}

main方法

main 是任务的实际执行内容。

默认情况下,NSOperationmain 方法是空的。开发者需要通过子类化 NSOperation 并重写 main 方法来定义任务的具体逻辑。

main 方法在调用 start 时被执行。一般来说main就是用来完成任务的具体逻辑,但一般来说,如果只是为了串行的完成相关任务,其实代码不需要放置在main方法之中,相关处理放在可见的控制器之中即可

重写NSOperation实现并发

了解了相关的方法和属性之后,其实如何让NSOperation实现并发其实也不算太难

@interface JCOperation : NSOperation
@end

@implementation JCOperation {
    BOOL _isExecuting;
    BOOL _isFinished;
}

- (instancetype)init {
    self = [super init];
    if (self) {
        _isExecuting = NO;
        _isFinished = NO;
    }
    return self;
}

- (void)start {
    if (self.isCancelled) {
        [self finish];
        return;
    }

    [self setIsExecuting:YES];

    // 异步操作开始
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self main];
    });
}

- (void)main {
    // 模拟异步任务(网络请求等)
    [NSThread sleepForTimeInterval:2];
    NSLog(@"任务完成 - %@", [NSThread currentThread]);

    [self finish];
}

- (void)finish {
    [self setIsExecuting:NO];
    [self setIsFinished:YES];
}

// 重写状态方法,getter方法
- (BOOL)isExecuting { return _isExecuting; }
- (BOOL)isFinished { return _isFinished; }

- (void)setIsExecuting:(BOOL)isExecuting {
  //手动触发KVO
    [self willChangeValueForKey:@"isExecuting"];
    _isExecuting = isExecuting;
    [self didChangeValueForKey:@"isExecuting"];
}

- (void)setIsFinished:(BOOL)isFinished {
    [self willChangeValueForKey:@"isFinished"];
    _isFinished = isFinished;
    [self didChangeValueForKey:@"isFinished"];
}
@end

在重写NSOperation子类当中,我们需要使用KVO来实现对应属性的改变,由于KVO 机制只能监听 属性,也就是类提供的 @property 定义的公开或受保护的接口。而在NSOperation之中其 isExecutingisFinished 属性是只读的,我们无法直接使用 self.executing = YES; 来修改其值。

  • 当我们自定义 NSOperation 子类时,_isExecuting_isFinished 这两个实例变量代替了原有的只读属性 isExecutingisFinished
  • 在内部逻辑中,我们直接操作这些实例变量来控制操作的状态。
  • 实际使用中,NSOperationQueue 或其他部分不直接访问这些变量,而是通过 isExecutingisFinishedgetter 方法获取状态。(这个很关键)

KVO的关键就在于通知其他部分属性的变化,通过 KVO,我们在修改实例变量时,将变化广播给 NSOperationQueue 等外部监听方:

  1. NSOperationQueue 使用 KVO 监听 isExecutingisFinished 的变化。
  2. isFinished 变为 YES 时,队列会认为操作已完成并从队列中移除,同时触发依赖任务的执行。

由于KVO 的工作原理要求使用完整的 getter 方法名称作为键路径,而不是属性名本身。所以我们使用KVO手动监听isExecutingisFinished

@property (readonly, getter=isExecuting) BOOL executing;
@property (readonly, getter=isFinished) BOOL finished;

简单的概括来说是:

实例变量是内部逻辑的真实值

KVO 是广播机制,通知外部(尤其是 NSOperationQueue)状态已经改变。

参考文章

NSOperation

彻底了解NSOperation的自定义

iOS Operation 自定义的注意点

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2256363.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

可供参考的GitHub国内镜像

在配置了本地hosts文件和魔法后仍存在无法访问的问题 针对如上问题&#xff0c;可以使用国内的镜像地址做替换 例如: https://github.com/bubbliiiing/detr-pytorch改成 https://hub.nuaa.cf/bubbliiiing/detr-pytorch推荐使用的镜像 https://hub.yzuu.cf/ https://hub.nua…

Codeforces Round 784 (Div. 4)

题目链接 A. Division? 题意 思路 模拟即可 示例代码 void solve() {int n;cin >> n;int ans;if(n > 1900) ans 1;else if(n > 1600) ans 2;else if(n > 1400) ans 3;else ans 4;cout << "Division " << ans << \n;}B. T…

E172 ASP.NET+SQL+C#+LW+图书管理系统的设计与实现 配置 源码 文档 全套资料

图书管理系统 1.项目摘要2. 系统的概述3.项目功能4.界面展示5.源码获取 1.项目摘要 摘 要 书籍是供人们获取并增长知识的主要途径&#xff0c;由于图书的种类较多&#xff0c;阅读者也较多&#xff0c;借阅量较大&#xff0c;且易出错&#xff0c;传统的图书借阅若还停留在手工…

TriCore架构-TC397将code从原来在P-Cache地址移到PSPR的地址,CPU的负载率为什么没影响

TC397有6个内核,每个核有自己的私有的Memory以及共有的Memory。 私有的:PSPR,DSPR,P-Cache,D-Cache,PF(X),LMU,DLMU,LPB PSPR主要用来运行RAM Code,比如说有些代码要放到RAM里面运行。 DSPR主要当成SRAM来用,比如用来存放全局变量。 P-Cache通过PFI接口访问DMU的3M内…

109.【C语言】数据结构之二叉树层序遍历

目录 1.知识回顾 2.代码实现 准备工作 LevelOrder函数 代码框架 关键代码 3.执行结果 1.知识回顾 层序遍历参见106.【C语言】数据结构之二叉树的三种递归遍历方式文章 截取的部分内容 定义:按层的方式遍历(,设n为树的深度,h1-->h2-->h3-->...-->hn) 以下面…

基于SpringBoot的养老院管理系统的设计与实现

一、前言 随着人口老龄化的加剧&#xff0c;养老院作为老年人养老的重要场所&#xff0c;其管理的高效性和科学性显得尤为重要。传统的养老院管理方式多依赖人工操作&#xff0c;存在信息记录不及时、不准确&#xff0c;管理流程繁琐&#xff0c;资源调配困难等问题。利用信息技…

012 路由信息协议RIP

路由信息协议RIP 作为度量(Metric)来衡量到达目的网络的距离 RIP是一种基于距离矢量D-V(Distance-Vector)算法的协议&#xff0c;它使用跳数(Hop Count)作为度量(Metric)来衡量到达目的网络的距离。 默认情况下&#xff0c;路由器到与它直接相连网络的跳数为0&#xff0c;因此…

NLP与LLM的工程化实践与学习思考 - 说说知识图谱

NLP与LLM的工程化实践与学习思考[24年半年工作总结] - 说说知识图谱 0 真的就是先说说1 为什么知识图谱什么是知识图谱&#xff1f;基于图的数据结构&#xff1f;基于数据结构的图&#xff1f;知识图谱的技术要点两个技术维度&#xff1a;知识、图七个技术要点&#xff1a;表示…

ROS2 - C++工程创建和工程解析

ROS2 系列文章目录 文章目录 ROS2 系列文章目录前言1. 安装构建工具colcon1.1 简介1.2 安装colcon 2. 创建工作空间2.1 创建一个工作目录 3 创建一个C软件包4. 编写发布者节点4.1 使用wget获取一个模板4.2 代码解读4.2 添加依赖项4.3 修改CmakeLists.txt 5. 编写订阅者节点5.1 …

【5G】架构 Architecture

5G网络架构受到了多个因素的影响。首先是为云端实现做好准备&#xff0c;其次是应对比之前更大的数据速率和更低的时延&#xff0c;启用新服务的能力&#xff0c;以及特别是在初期阶段与长期演进&#xff08;LTE&#xff09;的互操作需求。所有这些因素都对5G架构产生了影响。除…

vue2+html2canvas+js PDF实现试卷导出和打印功能

1.首先安装 import html2canvas from html2canvas; import { jsPDF } from jspdf; 2.引入打印插件print.js import Print from "/assets/js/print"; Vue.use(Print) // 打印类属性、方法定义 /* eslint-disable */ const Print function (dom, options) {if (…

图像滤波和卷积的不同及MATLAB应用实例

滤波与卷积在图像处理中都是非常重要的运算&#xff0c;但它们有着明显的区别。以下是滤波与卷积的主要不同点&#xff0c;并附带一个MATLAB实例来展示两者在图像处理中的效果差异。 一、滤波与卷积的不同 定义与目的&#xff1a; 1&#xff09;滤波&#xff1a;滤波是一种信…

Java集合(三)- Stack Queue

目录 一、Stack & Queue概述 1.1、Queue 1.2、Deque 二、方法剖析 2.1、addFirst() 2.2、addLast() 2.3、pollFirst() 2.4、pollLast() 2.5、peekFirst() 2.6、peekLast() 一、Stack & Queue概述 Java里有一个叫做Stack的类&#xff0c;却没有叫做Queue的类…

数据结构理论

内容来源青岛大学数据结构与算法课程&#xff0c;链接&#xff1a;数据结构与算法基础&#xff08;青岛大学-王卓&#xff09;_哔哩哔哩_bilibili 绪论 数据结构概述 数据结构和算法的定义&#xff1a;我们如何把现实中大量而复杂的问题以特定的数据类型和特定的存储结构保存…

008.精读《Apache Paimon Docs - Table w/o PK》

文章目录 1. 引言2. 基本概念2.1 定义2.2 使用场景 3. 流式处理3.1 自动小文件合并3.2 流式查询 4. 数据更新4.1 查询4.2 更新4.3 分桶附加表 5 总结 1. 引言 通过本文&#xff0c;上篇我们了解了Apache Paimon 主键表&#xff0c;本期我们将继续学习附加表&#xff08;Append…

硬件选型规则

光源选型: 先用型号中带H的&#xff0c;没有的选标准的. 光源和光源控制器的搭配需要确保接口一致。 根据型号表中的最佳工作距离和相机的尺寸。 光源控制器选型&#xff1a; 首先选择海康风格系列光源控制器考虑与光源的接口匹配。功率应该满足接近光源功率。检查是否退市…

数据分析系列---requests的使用

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” 主环境和虚拟环境 主环境是电脑上安装的python环境 虚拟环境在项目中可以实现环境的隔离&#xff0c;假设DemoA和DemoB分别用到了某个三方库1.0和2.0版本&#xff0c;那么在一个…

selenium常见接口函数使用

博客主页&#xff1a;花果山~程序猿-CSDN博客 文章分栏&#xff1a;测试_花果山~程序猿的博客-CSDN博客 关注我一起学习&#xff0c;一起进步&#xff0c;一起探索编程的无限可能吧&#xff01;让我们一起努力&#xff0c;一起成长&#xff01; 目录 1. 查找 查找方式 css_s…

深度学习之预备知识

深度学习是关于优化的学习。对一个带有参数的模型&#xff0c;找到其中能拟合数据最好的模型。 一、数据操作 张量&#xff1a;表示一个由数值组成的数组&#xff0c;这个数组可能有多个维度。具有一个轴的张量对应数学上的向量&#xff0c;具有两个轴的张量对应数学上的矩阵&…

【论文相关】期刊/会议 信息检索——IEEE各期刊投稿要求(待完善)

[吐槽] 外文期刊/会议网站的真的很难找&#xff0c;想要了解的信息很难检索&#xff1b;比如 旗下子期刊官网、子期刊具体要求不同等问题。 TIFS期刊 (IEEE Transactions on Information Forensics and Security) 期刊官网链接&#xff1a;https://mc.manuscriptcentral.com/…