OGeek|极客世界-中国程序员成长平台

标题: ios - 与 NSOperation 和 NSOperationQueue (Swift) 相互依赖的 NSURLSession 任务 [打印本页]

作者: 菜鸟教程小白    时间: 2022-12-13 13:51
标题: ios - 与 NSOperation 和 NSOperationQueue (Swift) 相互依赖的 NSURLSession 任务

问题: 我想在 Swift 中开发一个 iOS 应用程序,它在登录后立即执行初始加载。序列(通过 NSURLSession 的基于 REST 的调用)如下所示:

  1. 用用户账号登录 -> 异步响应返回userId
  2. 获取 userId 的国家 -> 异步响应返回 countryId
  3. 获取 countryId 的产品 -> ...等...

基本上我想找到一种优雅的方式来实现这样的序列。

方法: 首先,我只是在另一个的完成处理程序中调用新的(依赖的)REST 调用。但是如果需要执行很多调用并且依赖级别比上面描述的要多,那么代码看起来有点乱......

我在 WWDC 2015 上遇到了关于 NSOperations 的 session ,并认为这可能是一个好主意,因为有些人可以很容易地定义依赖关系。 不幸的是,Apple 提供的示例代码没有给出上述问题的答案……是吗(我没有得到它?)? 在玩操作操作时,我无法考虑在创建不同操作(LoginOperation、CountryOperation(依赖于 LoginOperation)、ProductOperation(依赖于 CountryOperation)等时如何解决初始化问题。)

我发现这些帖子很有帮助,但我仍然不了解如何最好地解决我描述的问题: How To Download Multiple Files Sequentially using NSURLSession downloadTask in Swift Chaining NSOperation : Pass result from an operation to the next one NSURLSession with NSBlockOperation and queues

困难: 在另一个操作 A 成功完成并返回将由操作 B 使用的结果时初始化操作 B。



Best Answer-推荐答案


如果仍然相关,我可能会有答案。

这里的问题是你试图推迟初始化直到第一个操作完成。这是一个死胡同。操作意味着要尽早创建,并且使用依赖项来保证顺序。

所以让我们讨论一些解决这个极其常见的问题的方法。

(您可以注意到这些示例是用 WWDC session 风格编写的。这实际上是因为我使用了基于该演讲的 Operations library。如果您对基于操作的架构感兴趣,请查看它。 )

1。使用一些外部可变共享状态(Cocoa-way)

这基本上意味着,对于您的示例,您有一些 UserController,您将其实例传递给 LoginOperationCountryOperation . LoginOperationuserId 写入该 Controller ,CountryOperation 从中读取:

class UserController {
    var userID: String?
}

class LoginOperation: Operation {

    let userController: UserController

    init(userController: UserController) {
        self.userController = userController
    }

    override func execute() {
        // do your networking stuff
        userController.userID = receivedUserID
        finish()
    }

}

class CountryOperation: Operation {

    let userController: UserController

    init(userController: UserController) {
        self.userController = userController
    }

    override func execute() {
        let userID = userController.userID
        // continue
    }

}

然后:

let queue = OperationQueue()
let userController = UserController()
let login = LoginOperation(userController: userController)
let country = CountryOperation(userController: userController)
country.addDependency(login)
queue.addOperation(login)
queue.addOperation(country)

该解决方案的问题在于它非常复杂且难以测试。而且 Swift 也不太喜欢可变共享状态。

2。使用函数而不是值来初始化操作(Swifty-way)

嗯,这不是显而易见但非常优雅的解决方案。如您所见,值在初始化时被复制。我们真正需要的是检索所需的执行时间值(与前面的示例一样),但没有如此紧密的耦合。好吧,这很容易做到 - 只需将一个函数传递给初始化程序!

class LoginOperation: Operation {

    private(set) var userID: String?

    override func execute() {
        // do your networking stuff
        let receivedUserID = "1"
        userID = receivedUserID
        finish()
    }

}

class CountryOperation: Operation {

    private let getUserID: () -> String

    init(getUserID: () -> String) {
        self.getUserID = getUserID
    }

    override func execute() {
        /* Because this operation depends on `LoginOperation`,
        `getUserID()` is called after `LoginOperation` is finished. */
        let userID = self.getUserID()
        // continue
    }

}

然后:

let queue = OperationQueue()
let login = LoginOperation()
// Force-unwrap is no good, of course, but for this example it's ok
let country = CountryOperation(getUserID: { login.userID! })
country.addDependency(login)
queue.addOperation(login)
queue.addOperation(country)

如您所见,没有紧密耦合,没有共享可变状态(CountryOperation 只能读取,不能写入——这很好)。而且测试起来非常简单:只需将您想要的任何内容传递给 CountryOperation 初始化程序,您根本不需要 LoginOperation 来测试它!

实际上,通过这种方法,NSOperation 的主要目标得以实现:独特的、抽象的工作。 CountryOperation 本身对 LoginOperation 以及任何其他共享实例一无所知。

我实际上在我的项目中经常使用这种方法,并且效果非常好。

如果您有任何其他问题,请回复

关于ios - 与 NSOperation 和 NSOperationQueue (Swift) 相互依赖的 NSURLSession 任务,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35035478/






欢迎光临 OGeek|极客世界-中国程序员成长平台 (https://ogeek.cn/) Powered by Discuz! X3.4