gpt4 book ai didi

swift - PromiseKit:在链的开头未使用时,委托(delegate)系统包装器似乎会立即返回

转载 作者:可可西里 更新时间:2023-11-01 02:16:59 24 4
gpt4 key购买 nike

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

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

下面是我的代码,其中:

  1. LoginToService 是基于 block 的方法的可 promise 版本,通过使用 PromiseKit 扩展 MCUuser 获得。这按预期工作并在用户登录后返回,否则会因错误而失败。

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

    我希望在后续的 .then 子句(以及最后的“finally”子句)被调用之前履行 promise - 当用户点击按钮关闭它时,警报的 promise 应该履行,按照PromiseKit 的委托(delegate)系统包装器的实现:当我使用 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 链的开头,我是否明显遗漏了某些东西或那些不打算使用的委托(delegate)系统包装器?

提前感谢任何提示

最佳答案

它应该如您所愿地工作。您可以检查返回的 promise 是否填写错误。

不过,让我提示的是 alert.promise()应该返回 Promise<Int> - 但闭包被显式键入以返回 AnyPromise .因此,您的代码不应编译。

我自己设置了一个测试项目,确实,编译器报错了。我使用了 PromiseKit v3.x。您的版本可能是旧版本(finallycatch 已弃用)。

将闭包的返回类型固定为 Promise<Int> 之后,代码编译。但重要的事实是,行为正如您在代码中所描述和体验的那样 - 而不是它应该的,恕我直言。所以,似乎有一个错误。

编辑:

OK,原来是“重载解析”和“类型推断”出了问题。鉴于您在 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 似乎存在一些微妙的陷阱。希望您现在可以解决您的问题。

关于swift - PromiseKit:在链的开头未使用时,委托(delegate)系统包装器似乎会立即返回,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37024195/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com