问题:
我想在 Swift 中开发一个 iOS 应用程序,它在登录后立即执行初始加载。序列(通过 NSURLSession 的基于 REST 的调用)如下所示:
- 用用户账号登录 -> 异步响应返回userId
- 获取 userId 的国家 -> 异步响应返回 countryId 的
- 获取 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-推荐答案 strong>
如果仍然相关,我可能会有答案。
这里的问题是你试图推迟初始化直到第一个操作完成。这是一个死胡同。操作意味着要尽早创建,并且使用依赖项来保证顺序。
所以让我们讨论一些解决这个极其常见的问题的方法。
(您可以注意到这些示例是用 WWDC session 风格编写的。这实际上是因为我使用了基于该演讲的 Operations library。如果您对基于操作的架构感兴趣,请查看它。 )
1。使用一些外部可变共享状态(Cocoa-way)
这基本上意味着,对于您的示例,您有一些 UserController ,您将其实例传递给 LoginOperation 和 CountryOperation . LoginOperation 将 userId 写入该 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/
|