首页 >后端开发 >C++ >为什么我的异步 C# 方法在访问任务结果时挂起?

为什么我的异步 C# 方法在访问任务结果时挂起?

Barbara Streisand
Barbara Streisand原创
2025-01-08 13:47:42991浏览

Why Does My Asynchronous C# Method Hang When Accessing Task Results?

C#异步方法在访问Task结果时挂起的原因

使用C#的asyncawait关键字进行异步编程时,某些结构可能会导致意外行为和潜在的死锁。

考虑以下场景:一个多层应用程序使用扩展的数据库实用程序方法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的使用,并将ExecuteAsyncGetTotalAsync方法重写为不等待的纯异步方法:

<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中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn