PromiseKit:委托系统包装器似乎在未在链的开头使用时立即返回

时间:2016-05-04 09:50:24

标签: swift promisekit

我是PromiseKit的新手,我已经尝试了几天来找出一个解决方案,以解决promise-wrapped委托系统的意外行为(UIALertView + PromiseKit,PMKLocationManager等。 )。

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

以下是我的代码,其中:

  1. LoginToService是通过使用PromiseKit扩展MCUser获得的基于块的方法的可承诺版本。这可以按预期工作,并在用户登录后返回,或者因错误而失败。
  2. 在'然后'成功登录的子句,我通过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")
    }
    
  3. 我观察到的是,在没有等待用户点击的情况下,链条会立即继续,并且点击了'和'最后'消息被打印到控制台,屏幕上的警报等待操作。我是否明显遗漏了某些东西,或者如果不是在Promise链的开头,那些代理系统包装器不应该被使用?

    提前感谢任何提示

1 个答案:

答案 0 :(得分:1)

它应该按预期工作。您可以检查返回的承诺是否未正确完成。

让我抱怨的是,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似乎存在一些微妙的陷阱。希望你现在可以解决你的问题。