Maison >développement back-end >C++ >Pourquoi ma méthode C# asynchrone se bloque-t-elle lors de l'accès aux résultats des tâches ?

Pourquoi ma méthode C# asynchrone se bloque-t-elle lors de l'accès aux résultats des tâches ?

Barbara Streisand
Barbara Streisandoriginal
2025-01-08 13:47:42951parcourir

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

La raison pour laquelle la méthode asynchrone C# se bloque lors de l'accès aux résultats de la tâche

Lors de l'utilisation des mots-clés async et await de C# pour la programmation asynchrone, certaines constructions peuvent provoquer un comportement inattendu et des blocages potentiels.

Considérez le scénario suivant : une application multiniveau utilise une méthode utilitaire de base de données étendue ExecuteAsync qui exécute de manière asynchrone une requête SQL et renvoie les résultats. La méthode de couche intermédiaire GetTotalAsync appelle ExecuteAsync pour récupérer les données et stocke le résultat dans la variable asyncTask. Enfin, les opérations de l'interface utilisateur tentent d'accéder aux résultats de manière synchrone à l'aide de asyncTask.Result. Cependant, l'application se bloque indéfiniment.

Cause de l'impasse

Le problème vient de l'utilisation de GetTotalAsync dans la méthode await. Par défaut, les suites des méthodes asynchrones sont distribuées sur le même SynchronizationContext qui a démarré la méthode. Dans ce cas, lors de l'utilisation de await sur le thread UI, la suite (return result;) est également programmée pour s'exécuter sur le thread UI.

Lorsque asyncTask.Result est appelé sur le thread de l'interface utilisateur, il bloque le thread une fois la tâche terminée. Cependant, les continuations planifiées sur le thread de l'interface utilisateur ne peuvent pas s'exécuter tant que asyncTask.Result n'est pas terminé. Cela crée une impasse dans laquelle aucun des threads ne peut continuer l'exécution.

Solution

Afin de sortir de cette impasse, il existe plusieurs méthodes :

1. Supprimer le mot-clé asynchrone :

Éliminer l'utilisation de await et réécrire les méthodes ExecuteAsync et GetTotalAsync comme des méthodes purement asynchrones qui n'attendent pas :

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

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

2. Utilisez ConfigureAwait :

Utilisez ConfigureAwait(false) pour spécifier que les continuations ne doivent pas être planifiées sur le fil de discussion de l'interface utilisateur :

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

Notez que cette approche nécessite une spécification explicite de await sur toutes les ConfigureAwait(false) opérations pouvant conduire à une impasse.

3. Utilisez SynchronizationContext :

Créez un SynchronizationContext spécifique pour les opérations asynchrones et assurez-vous que toutes les opérations await utilisent ce contexte, évitant ainsi les conflits avec le thread de l'interface utilisateur.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn