Heim >Backend-Entwicklung >C++ >Warum bleibt meine asynchrone C#-Methode beim Zugriff auf Aufgabenergebnisse hängen?

Warum bleibt meine asynchrone C#-Methode beim Zugriff auf Aufgabenergebnisse hängen?

Barbara Streisand
Barbara StreisandOriginal
2025-01-08 13:47:42950Durchsuche

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

Der Grund, warum die asynchrone C#-Methode beim Zugriff auf Aufgabenergebnisse hängt

Bei Verwendung der C#-Schlüsselwörter async und await für die asynchrone Programmierung können bestimmte Konstrukte unerwartetes Verhalten und potenzielle Deadlocks verursachen.

Stellen Sie sich das folgende Szenario vor: Eine mehrschichtige Anwendung verwendet eine erweiterte Datenbankdienstprogrammmethode ExecuteAsync, die asynchron eine SQL-Abfrage ausführt und die Ergebnisse zurückgibt. Die Mittelschichtmethode GetTotalAsync ruft ExecuteAsync auf, um die Daten abzurufen, und speichert das Ergebnis in der Variablen asyncTask. Schließlich versuchen UI-Operationen, mithilfe von asyncTask.Result synchron auf die Ergebnisse zuzugreifen. Die Anwendung bleibt jedoch auf unbestimmte Zeit hängen.

Ursache des Deadlocks

Das Problem entsteht durch die Verwendung von GetTotalAsync in der Methode await. Standardmäßig werden Fortsetzungen asynchroner Methoden auf demselben SynchronizationContext ausgeführt, der die Methode gestartet hat. In diesem Fall ist bei Verwendung von await im UI-Thread die Ausführung der Fortsetzung (return result;) auch im UI-Thread geplant.

Wenn asyncTask.Result im UI-Thread aufgerufen wird, blockiert es den Thread, wenn die Aufgabe abgeschlossen ist. Allerdings können im UI-Thread geplante Fortsetzungen erst ausgeführt werden, wenn asyncTask.Result abgeschlossen ist. Dadurch entsteht ein Deadlock, bei dem keiner der Threads die Ausführung fortsetzen kann.

Lösung

Um diesen Deadlock zu lösen, gibt es mehrere Methoden:

1. Async-Schlüsselwort löschen:

Beseitigen Sie die Verwendung von await und schreiben Sie die Methoden ExecuteAsync und GetTotalAsync als reine asynchrone Methoden um, die nicht warten:

<code class="language-csharp">public static Task<T> ExecuteAsync<T>(this OurDBConn dataSource, Func<OurDBConn, T> function)
{
    // ... (代码保持不变)
}

public static Task<ResultClass> GetTotalAsync(...)
{
    // ... (代码保持不变)
}</code>

2. Verwenden Sie „ConfigureAwait:“

Verwenden Sie ConfigureAwait(false), um anzugeben, dass Fortsetzungen nicht im UI-Thread geplant werden sollen:

<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>

Beachten Sie, dass dieser Ansatz eine explizite Angabe von await für alle ConfigureAwait(false)-Vorgänge erfordert, die zu einem Deadlock führen können.

3. Verwenden Sie SynchronizationContext:

Erstellen Sie ein spezifisches SynchronizationContext für asynchrone Vorgänge und stellen Sie sicher, dass alle await-Vorgänge diesen Kontext verwenden, um Konflikte mit dem UI-Thread zu vermeiden.

Das obige ist der detaillierte Inhalt vonWarum bleibt meine asynchrone C#-Methode beim Zugriff auf Aufgabenergebnisse hängen?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn