2

PromiseKit还很陌生,而且我已经尝试了几天来找出一个解决方案,以解决承诺包装的委托系统(UIALertView+PromiseKitPMKLocationManager等)的意外行为。

在我相当典型的应用程序设置过程场景中,我试图链接用户在应用程序加载时必须执行的一系列操作。为了这个例子,让我们将案例限制为只有两个步骤:将用户登录到一个 Restful 系统,然后显示一个 alertView 并等待用户的交互。

下面是我的代码,其中:

  1. LoginToService 是通过扩展 MCUser 获得的基于块的方法的可承诺版本PromiseKit。这可以按预期工作,并在用户登录后返回,或者因错误而失败。

  2. 在成功登录的“then”子句中,我通过 alert.promise() 返回其承诺版本来呈现一个 alertView。

    我希望在调用连续的 .then 子句(最后是“finally”子句)之前实现该承诺 - 根据 PromiseKit 的实现,当用户单击按钮以将其关闭时,应该实现警报的承诺委托系统包装器:当我使用 alert.promise().then 启动 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 链的开头,我是否显然遗漏了某些东西,或者那些不打算使用的委托系统包装器?

提前感谢您的任何提示

4

1 回答 1

1

它应该按您的预期工作。您可以检查返回的 Promise 是否错误完成。

然而,让我抱怨的是,它alert.promise()应该返回 a Promise<Int>- 但闭包被显式键入以返回 a 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>

这是由继承 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 似乎存在一些微妙的缺陷。希望您现在可以解决您的问题。

于 2016-05-04T10:08:22.993 回答