我是PromiseKit
的新手,我已经尝试了几天来找出一个解决方案,以解决promise-wrapped委托系统的意外行为(UIALertView + PromiseKit,PMKLocationManager
等。 )。
在我相当典型的应用程序安装过程中,我试图将用户在加载应用程序时必须经历的一系列操作链接起来。 为了这个例子,让我们将案例限制为两个步骤: 将用户登录到Restful系统,然后显示alertView并等待用户的交互。
以下是我的代码,其中:
PromiseKit
扩展MCUser获得的基于块的方法的可承诺版本。这可以按预期工作,并在用户登录后返回,或者因错误而失败。在'然后'成功登录的子句,我通过alert.promise()返回其承诺版本来呈现alertView。
我希望在连续的.then子句(以及最后的' finally'子句)被调用之前履行承诺 - 当用户应该满足警告的承诺根据PromiseKit的委托系统包装器的实现,单击按钮来关闭它:当我使用alert.promise()时,这对于观察到的行为很好。然后启动Promises链 -
// Doesn't work: alert.promise returns immediately
let user = MCUser.sharedInstance()
user.logInToService(.RestServiceOne, delegate: self).then { _ -> AnyPromise in
MCLogInfo("LogInToService: Promise fulfilled")
let alert = UIAlertView(title: "Title", message: "Msg", delegate: nil, cancelButtonTitle: "Cancel", otherButtonTitles: "Hello")
return alert.promise()
}.then { (obj:AnyObject) -> Void in
print("Clicked")
}.finally {
print("Finally")
}.catch_ { error in
print("Error")
}
我观察到的是,在没有等待用户点击的情况下,链条会立即继续,并且点击了'和'最后'消息被打印到控制台,屏幕上的警报等待操作。我是否明显遗漏了某些东西,或者如果不是在Promise链的开头,那些代理系统包装器不应该被使用?
提前感谢任何提示
答案 0 :(得分:1)
它应该按预期工作。您可以检查返回的承诺是否未正确完成。
让我抱怨的是,alert.promise()
应该返回Promise<Int>
- 但明确键入闭包以返回AnyPromise
。所以,你的代码不应该编译。
我自己设置了一个测试项目,实际上,编译器抱怨。我使用了PromiseKit v3.x.您的版本可能是旧版本(finally
和catch
已弃用)。
将闭包的返回类型修复为Promise<Int>
后,代码会编译。但重要的事实是,行为就像你在代码中描述和体验的那样 - 而不是它应该如此,恕我直言。所以,似乎有一个错误。
修改强>
好的,事实证明“重载解析”和“类型推断”存在问题。鉴于你的OP中的代码,Swift编译器解决了then
方法的意外重载:
预期:
func then<U>(on q: dispatch_queue_t = dispatch_get_main_queue(), _ body: (T) throws -> Promise<U>) -> Promise<U>
实际:
func then<U>(on q: dispatch_queue_t = dispatch_get_main_queue(), _ body: (T) throws -> U) -> Promise<U>
这是由后续 finally
和catch
方法引起的。
为了在这种情况下解决它,您应该正确完全指定闭包的类型,或者让编译器通过不指定类型来自行计算出来。我终于得到了这个,使用PromiseKit v3.x和Swift按预期工作:
import UIKit
import PromiseKit
// helper
func fooAsync() -> Promise<String> {
return Promise { fulfill, reject in
let delay = dispatch_time(DISPATCH_TIME_NOW, Int64(1.0 * Double(NSEC_PER_SEC)))
dispatch_after(delay, dispatch_get_global_queue(0,0)) {
fulfill("OK")
}
}
}
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
fooAsync()
.then { str in
print("String: \(str)")
let alert = UIAlertView(title: "Title", message: "Msg", delegate: nil, cancelButtonTitle: "Cancel", otherButtonTitles: "Hello")
let promise: Promise<Int> = alert.promise()
return promise
}.then { (n: Int) -> Void in // notice: closure type specified!
print("Clicked")
}.ensure {
print("Finally")
}.report { error in
print("Error")
}
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
}
以上代码可能无法解决您的问题,因为您使用的是其他PromiseKit库。我建议使用最新的Swift版本。
尽管如此,PromiseKit似乎存在一些微妙的陷阱。希望你现在可以解决你的问题。