C#异步方法在访问Task结果时挂起的原因
使用C#的async
和await
关键字进行异步编程时,某些结构可能会导致意外行为和潜在的死锁。
考虑以下场景:一个多层应用程序使用扩展的数据库实用程序方法ExecuteAsync
,该方法异步执行SQL查询并返回结果。中间层方法GetTotalAsync
调用ExecuteAsync
来检索数据并将结果存储在asyncTask
变量中。最后,UI操作尝试使用asyncTask.Result
同步访问结果。但是,应用程序无限期挂起。
死锁的原因
问题源于在GetTotalAsync
方法中使用await
。默认情况下,异步方法的延续在启动方法的同一SynchronizationContext
上调度。在这种情况下,当在UI线程上使用await
时,延续(return result;
)也计划在UI线程上运行。
当在UI线程上调用asyncTask.Result
时,它会在Task完成时阻塞线程。但是,在UI线程上调度的延续直到asyncTask.Result
完成才能执行。这会创建一个死锁,两个线程都无法继续执行。
解决方案
为了解决这个死锁,有几种方法:
1. 删除Async关键字:
消除await
的使用,并将ExecuteAsync
和GetTotalAsync
方法重写为不等待的纯异步方法:
<code class="language-csharp">public static Task<T> ExecuteAsync<T>(this OurDBConn dataSource, Func<OurDBConn, T> function) { // ... (代码保持不变) } public static Task<ResultClass> GetTotalAsync(...) { // ... (代码保持不变) }</code>
2. 使用ConfigureAwait:
使用ConfigureAwait(false)
指定延续不应在UI线程上调度:
<code class="language-csharp">public static async Task<ResultClass> GetTotalAsync(...) { var resultTask = this.DBConnection.ExecuteAsync<ResultClass>( ds => ds.Execute("select slow running data into result")); return await resultTask.ConfigureAwait(false); }</code>
注意,这种方法需要在所有可能导致死锁的await
操作上显式指定ConfigureAwait(false)
。
3. 使用SynchronizationContext:
为异步操作创建一个特定的SynchronizationContext
,并确保所有await
操作都使用该上下文,从而防止与UI线程发生冲突。
以上是为什么我的异步 C# 方法在访问任务结果时挂起?的详细内容。更多信息请关注PHP中文网其他相关文章!