- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
正如我有限(甚至错误)的理解,Async.StartImmediate 和 Async.RunSynchronously 在当前线程上启动异步计算。那么这两个功能究竟有什么区别呢?谁能帮忙解释一下?
更新:
在 https://github.com/fsharp/fsharp/blob/master/src/fsharp/FSharp.Core/control.fs 查看 F# 源代码后,我想我有点明白会发生什么。 Async.StartImmediate 在当前线程上启动异步。在它命中异步绑定(bind)之后,它是否会继续在当前线程上运行取决于异步绑定(bind)本身。例如,如果异步绑定(bind)调用 Async.SwitchToThreadPool,它将在 ThreadPool 而不是当前线程上运行。在这种情况下,如果您想返回当前线程,则需要调用 Async.SwitchToContext。否则,如果异步绑定(bind)没有做任何切换到其他线程,Async.StartImmediate 将继续在当前线程上执行异步绑定(bind)。在这种情况下,如果您只想停留在当前线程上,则无需调用 Async.SwitchToContext。
Dax Fohl 的示例在 GUI 线程上工作的原因是因为 Async.Sleep 仔细捕获
SynchronizationContext.Current 并确保继续在捕获的上下文中运行,使用
SynchronizationContext.Post()。见 https://github.com/fsharp/fsharp/blob/master/src/fsharp/FSharp.Core/control.fs#L1631 ,其中 unprotectedPrimitiveWithResync 包装器更改了“args.cont”(继续)
成为捕获上下文的 Post(参见:https://github.com/fsharp/fsharp/blob/master/src/fsharp/FSharp.Core/control.fs#L1008 — trampolineHolder.Post 基本上是 SynchronizationContext.Post)。这只会工作
当 SynchronizationContext.Current 不为空时,GUI 线程总是如此。尤其,
如果你在控制台应用中使用 StartImmediate 运行,你会发现 Async.Sleep 确实会进入 ThreadPool,因为控制台应用中的主线程没有 SynchronizationContext.Current。
总而言之,这确实适用于 GUI 线程,因为某些函数(如 Async.Sleep、Async.AwaitWaitHandle 等)会仔细捕获并确保回发到先前的上下文。
看起来这是一种故意的行为,但是这似乎没有在 MSDN 中的任何地方记录。
最佳答案
Async.RunSynchronously
等到整个计算完成。因此,当您需要从常规代码运行异步计算并需要等待结果时,请使用它。很简单。Async.StartImmediate
确保计算在当前上下文中运行,但不会等到整个表达式完成。最常见的用途(至少对我而言)是当您想在 GUI 线程上异步运行计算时。例如,如果你想以 1 秒的间隔在 GUI 线程上做三件事,你可以写
async {
do! Async.Sleep 1000
doThing1()
do! Async.Sleep 1000
doThing2()
do! Async.Sleep 1000
doThing3()
} |> Async.StartImmediate
RunSynchronously
在那里,它会在一段时间内阻塞 GUI 线程,并且您的屏幕将变得无响应。
// Async.StartImmediate
async {
printfn "Running"
do! Async.Sleep 1000
printfn "Finished"
} |> Async.StartImmediate
printfn "Next"
> Running
> Next
// 1 sec later
> Finished
// Async.RunSynchronously
async {
printfn "Running"
do! Async.Sleep 1000
printfn "Finished"
} |> Async.RunSynchronously
printfn "Next"
> Running
// 1 sec later
> Finished
> Next
// Async.Start just for completion:
async {
printfn "Running"
do! Async.Sleep 1000
printfn "Finished"
} |> Async.Start
printfn "Next"
> Next
> Running // With possible race condition since they're two different threads.
// 1 sec later
> Finished
Async.StartImmediate
不能返回值(因为在继续之前它没有运行完成),而
RunSynchronously
能够。
关于f# - Async.StartImmediate vs Async.RunSynchronously,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39546905/
正如我有限(甚至错误)的理解,Async.StartImmediate 和 Async.RunSynchronously 在当前线程上启动异步计算。那么这两个功能究竟有什么区别呢?谁能帮忙解释一下?
我继承了一个代码,它使用 NSURLConnection 和 initWithRequest:delegate:startImmediately:,因此通过应用程序委托(delegate)接收响应数据
我是一名优秀的程序员,十分优秀!