gpt4 book ai didi

ios - 与 NSOperation 和 NSOperationQueue (Swift) 相互依赖的 NSURLSession 任务

转载 作者:塔克拉玛干 更新时间:2023-11-02 21:08:39 25 4
gpt4 key购买 nike

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

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

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

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

我参加了关于 NSOperations 的 WWDC 2015 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。

最佳答案

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

这里的问题是您试图将初始化推迟到第一个操作完成。那是死路一条。操作旨在尽早创建,并使用依赖项保证顺序。

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

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

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

这基本上意味着,对于您的示例,您有一些 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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com