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 【问题描述】:

我对@9​​87654321@ 还很陌生,我已经尝试了几天来找出一个解决方案,以解决承诺包装的委托系统(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。您的可能是旧版本(finallycatch 已弃用)。

将闭包的返回类型固定为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>

这是由成功 finallycatch 方法引起的。

为了在这种情况下解决它,您应该正确完全指定闭包的类型,或者让编译器自己找出它而不指定类型。我终于得到了这个,它使用 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:委托系统包装器似乎在链的开头未使用时立即返回的主要内容,如果未能解决你的问题,请参考以下文章

PromiseKit 无法在链中命中

从 PromiseKit 包装迁移

r data.table 围绕 ad-hoc 连接的函数包装器(在链中聚合)

python委托(包装器)

Xamarin IOS LexDB Save - 尝试 JIT 编译的异常(包装器委托调用)

PromiseKit入门