3

idToken在返回变量之前,我正在等待响应。

请不要告诉我只使用完成处理程序并在没有DispatchGroup. 我知道我可以做到,但我试图理解为什么这个逻辑不起作用。

func createToken() -> String {
    
    var token = "empty"
    var group = DispatchGroup()
    group.enter()
    
    let currentUser = Auth.auth().currentUser
    
    currentUser?.getIDTokenForcingRefresh(true) { idToken, error in
        
        token = idToken ?? "error"
        print("Token Set")
        group.leave()
    }
    
    group.wait(timeout: DispatchTime.now() + 10)

    return token
}

跑步:

print("create ")
print(createToken())
print("done")

输出:

create 
empty
done
Token Set
4

1 回答 1

4

如果getIDTokenForcingRefresh将其完成处理程序闭包分派回主队列,您将准确地看到您描述的行为。当执行命中wait调用时,它将阻塞主线程。wait但是如果将完成处理程序闭包分派到主队列,则在调用超时并且再次释放主线程之前,它将无法运行。

您可以通过多种方式确认这个论点,即getIDTokenForcingRefresh在主线程上调用其完成处理程序:

  • 尝试删除timeout,在这种情况下,如果使用主队列/线程,此代码将完全死锁。

  • 打印Thread.isMainThread

  • 添加前置条件语句:

    dispatchPrecondition(condition: .onQueue(.main))
    
  • 在这个闭包中设置一个断点,让调试器告诉你使用了哪个线程。

但你说得很对:通常wait应该避免这种模式。

于 2020-11-10T23:39:02.840 回答