PromiseKit:委托系统包装器似乎在链的开头未使用时立即返回
Posted
技术标签:
【中文标题】PromiseKit:委托系统包装器似乎在链的开头未使用时立即返回【英文标题】:PromiseKit: delegate-system wrappers seem to return immediately when not used at the beginning of the chain 【发布时间】:2016-05-04 09:50:24 【问题描述】:我对@987654321@ 还很陌生,我已经尝试了几天来找出一个解决方案,以解决承诺包装的委托系统(UIALertView+PromiseKit、PMKLocationManager
等)的意外行为。
在我相当典型的应用设置过程场景中,我试图将应用加载时用户必须执行的一系列操作链接起来。 为了这个例子,让我们将案例限制为只有两个步骤: 将用户登录到 Restful 系统,然后显示 alertView 并等待用户交互。
以下是我的代码,其中:
LoginToService 是通过使用PromiseKit
扩展 MCUser 获得的基于块的方法的可承诺版本。这可以按预期工作,并在用户登录后返回,或者因错误而失败。
在成功登录的 'then' 子句中,我通过 alert.promise() 返回其承诺版本来呈现一个 alertView。
我希望在调用连续的 .then 子句(最后是“finally”子句)之前履行承诺 - 当用户单击按钮将其关闭时,警报的承诺应该履行,按照PromiseKit 的委托系统包装器的实现:当我使用 alert.promise().then 启动 Promise 链时,这可以很好地观察到的行为 -
// 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 链的开头,我是否显然遗漏了某些东西,或者那些委托系统包装器不应该被使用?
提前感谢任何提示
【问题讨论】:
【参考方案1】:它应该可以按您的预期工作。您可以检查返回的 Promise 是否错误完成。
不过,让我感到不满的是,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 似乎存在一些微妙的缺陷。希望您现在可以解决您的问题。
【讨论】:
感谢您的回复,很高兴听到这不是预期的行为。我在这里使用 UIAlertView 的示例,因为那是所有 PromiseKit 代码,因此其承诺的完成不应依赖于我的最终错误。在这种情况下,alert.promise() 在 promise() 方法实现结束后立即返回,从而将链继续到最后,当用户在稍后阶段单击按钮时我们也会返回。 @MattewUK 我做了一些额外的 cmets,请检查。 我在上面省略的一个细节是我正在使用它的 Obj-C 版本(UIAlertView+PromiseKit),其中 promise() 返回 AnyPromise,因此它可以编译并运行。我还使用它的 Swift 版本设置了一个测试项目,但正如你指出的那样,我得到了同样的奇怪行为。感谢您抽出宝贵时间仔细检查,我可能需要继续提出问题。 @MattewUK 最后,我找到了问题并得到了一个工作版本。请查看我更新的帖子。 那里做得很好,我非常感谢你发现这一点。实际上,即使在我使用的库版本中,通过切换到 Swift 实现来使用完全指定的返回类型似乎也可以正常工作。我将编辑错误报告并让他们深入研究这个问题,因为它似乎可能会影响其他情况。如果您想分享更多的解决过程以及您最终如何找到重载解决/类型推断问题......好吧,这将是一本非常有趣的读物。赞赏。以上是关于PromiseKit:委托系统包装器似乎在链的开头未使用时立即返回的主要内容,如果未能解决你的问题,请参考以下文章
r data.table 围绕 ad-hoc 连接的函数包装器(在链中聚合)