public string SomeSyncOperation(int someArg) { // sync code SomeAsyncOperation(someArg,someOtherArg).ConfigureAwait(false).GetAwaiter().GetResult() // sync code };
这里我们有一些必须访问异步api的同步代码,因此它会阻塞,直到结果准备就绪.我们不能在这里改变签名方法并添加异步.所以,我们无论如何都在等待,所以我们在这里需要ConfigureAwait(false)吗?我很确定我们没有,但我有点害怕删除它,因为它可能涵盖了一些用例(或者为什么我几乎到处都看到它?这只是一个货物崇拜?)并且删除此调用可能导致一些不安全的结果.
那么它有意义吗?
解决方法
How to correctly block on async code?
您没有正确阻止异步代码.阻止是错误的.询问做错事的正确方法是非首发.
>我手头有一个表示异步操作的对象.
>异步操作本身异步等待第二次异步操作完成.
>当消息循环执行与此线程的消息队列中当前消息相关联的代码时,将对此线程调度第二个异步操作.
现在,当您尝试同步获取第一个异步操作的结果时,您可以找出可怕的错误.它会阻塞,直到它的子异步操作完成,这将永远不会发生,因为现在我们已经阻止了将来为该请求提供服务的线程!
你的选择是:
>使整个调用堆栈正确异步并等待结果.
>不要使用此API.从头开始编写一个您知道不会死锁的等效同步API,并正确调用它.
>写一个不正确的程序,有时会无法预测死锁.
有两种方法可以编写正确的程序;在异步函数上编写同步包装是危险和错误的.
现在,您可能会问,ConfigureAwait是否通过删除我们在当前上下文中恢复的要求来解决问题?这不是我们担心的恢复点.如果您将依赖ConfigureAwait来保存死锁,那么堆栈中的每个异步操作都必须使用它,而我们不知道即将导致死锁的底层异步操作是否会这样做!
如果上述内容对您来说并不完全清楚,请阅读斯蒂芬关于为什么这是一个不好的做法的文章,以及为什么常见的解决方法只是危险的黑客攻击.
https://blog.stephencleary.com/2012/07/dont-block-on-async-code.html
https://msdn.microsoft.com/en-us/magazine/mt238404.aspx?f=255&MSPPError=-2147217396
但同样重要的是:正确的做法是重新设计你的程序以拥抱异步并在整个过程中使用等待.不要试图解决它.
becuase this method has stacktrace of ~20 methods,some of them are implementing some interfaces. Changing it to be async require change declarations in ~50 files,and we convert fully sync interfaces to mixed ones.
然后忙!听起来很简单.