Swift 中的 Operation 和 OperationQueue 如何使用

Operation 在 Swift 中是一种非常好用的类,它可以根据职责将业务拆分成多个类,保持代码的易读性。同时它还可以跟踪任务执行进度和控制任务依赖关系。

它和 ObjC 中 NSOperation 是一样的,一般结合 OperationQueue 来使用。

Operation 详解

在日常开发中,一个 Operation 的实例通常负责单个同步任务。需要注意的是:Operation 是一个抽象类,我们不应该直接使用它。而是应该使用系统实现的子类 BlockOperation 或者自己集成它,来自定义实现一个子类。

Operation 的任务启动方式有两种:

  • 将其添加到 OperationQueue
  • 手动调用其 start 方法启动

推荐 由 OperationQueue 来管理状态。

使用系统定义的 BlockOperation 代码示例如下:

scss 复制代码
let blockOperation = BlockOperation { 
    print("执行任务")
} 
let queue = OperationQueue() queue.addOperation(blockOperation)

当然,也可以直接调用 OperationQueue 的方法来添加任务:

arduino 复制代码
queue.addOperation { 
    print("执行任务")
}

给定的任务被添加到 OperationQueue中,它将尽快开始执行。

如何自定义 Operation 的子类

你可以通过自定义实现其子类,来进行业务代码的解耦。比如你有一个需求:导入用户的数据并将其上传到服务器。你可以实现一个负责导入的 Operation、再创建一个负责上传的 Operation。

下面的代码示例展示了一个用于导入内容的自定义子类:

swift 复制代码
final class ContentImportOperation: Operation {

    let itemProvider: NSItemProvider

    init(itemProvider: NSItemProvider) {
        self.itemProvider = itemProvider
        super.init()
    }

    override func main() {
        guard !isCancelled else { return }
        print("数据正在导入")
    }根据操作当前的执行状态,操作可以处于几种状态
}

该类接受一个 itemProvider 的参数来接受导入的内容,并在 main 方法中将其导入。main 方法是唯一需要覆盖同步操作的方法。

将操作添加到队列中,并设置一个 block 来跟踪完成:

ini 复制代码
let fileURL = URL(fileURLWithPath: "xxxx")
let contentImportOperation = ContentImportOperation(itemProvider: NSItemProvider(contentsOf: fileURL)!)

contentImportOperation.completionBlock = {
    print("导入完成")
}

queue.addOperation(contentImportOperation)

通过自定义 ContentImportOperation,我们可以将导入内容相关的逻辑都放在其中,在该类中跟踪任务进度和完成情况,这样代码逻辑会很清晰,并且测试代码也非常好写。

Operation 的几种状态

根据 Operation 当前的执行状态,其可以处于几种状态:

  • Ready:准备好开始任务
  • Executing:任务正在执行中
  • Finished:任务已经完成
  • Canceled:任务已被取消

需要注意的是,一个 Operation 只能执行一次。无论何时它处于完成或取消状态,你都不能再重新启动同一个实例。

在自定义实现中,你需要在执行之前手动检查取消状态,以确保任务被取消。还有就是,当 Operation 在同一时间开始和取消时,可能会发生数据竞争的问题。

一旦任务完成,OperationQueue 就会自动从队列中删除任务,不仅仅只有执行完成才会删除任务,取消任务也会触发。

管理依赖

使用 Operation 的一个很大的好处是使用依赖关系。你可以轻松地在两个实例之间添加依赖项。例如:导入内容后开始上传:

ini 复制代码
let fileURL = URL(fileURLWithPath: "xxx")
let contentImportOperation = ContentImportOperation(itemProvider: NSItemProvider(contentsOf: fileURL)!)
contentImportOperation.completionBlock = {
    print("导入完成")
}

let contentUploadOperation = UploadContentOperation()
contentUploadOperation.addDependency(contentImportOperation)
contentUploadOperation.completionBlock = {
    print("上传完成")
}

queue.addOperations([contentImportOperation, contentUploadOperation], waitUntilFinished: true)

只有在内容导入完成后,上传任务才会开始执行。需要说一下的是:如果导入操作取消,上传仍将开始。因为对于 Operation 来说它并不关心任务是执行导致完成还是取消导致的完成。

所以如果你想取消导入并不触发上传的话,你还需要在上传中添加判断逻辑:

swift 复制代码
final class UploadContentOperation: Operation {
    override func main() {
        guard !dependencies.contains(where: { $0.isCancelled }), !isCancelled else {
            return
        }

        print("上传中")
    }
}
相关推荐
Magnetic_h17 小时前
【iOS】单例模式
笔记·学习·ui·ios·单例模式·objective-c
归辞...18 小时前
「iOS」——单例模式
ios·单例模式·cocoa
yanling202320 小时前
黑神话悟空mac可以玩吗
macos·ios·crossove·crossove24
归辞...1 天前
「iOS」viewController的生命周期
ios·cocoa·xcode
crasowas1 天前
Flutter问题记录 - 适配Xcode 16和iOS 18
flutter·ios·xcode
2401_852403551 天前
Mac导入iPhone的照片怎么删除?快速方法讲解
macos·ios·iphone
SchneeDuan1 天前
iOS六大设计原则&&设计模式
ios·设计模式·cocoa·设计原则
JohnsonXin2 天前
【兼容性记录】video标签在 IOS 和 安卓中的问题
android·前端·css·ios·h5·兼容性
蒙娜丽宁2 天前
Go语言错误处理详解
ios·golang·go·xcode·go1.19
名字不要太长 像我这样就好2 天前
【iOS】push和pop、present和dismiss
学习·macos·ios·objective-c·cocoa