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
Libérer: 2025-01-08 13:47:42
original
1008 Les gens l'ont consulté

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 :

public static Task<T> ExecuteAsync<T>(this OurDBConn dataSource, Func<OurDBConn, T> function)
{
    // ... (代码保持不变)
}

public static Task<ResultClass> GetTotalAsync(...)
{
    // ... (代码保持不变)
}
Copier après la connexion

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 :

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);
}
Copier après la connexion

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 de ce site Web
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
Derniers articles par auteur
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal