IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Windows Presentation Foundation Discussion :

Multithread en WPF


Sujet :

Windows Presentation Foundation

  1. #1
    Membre confirmé
    Inscrit en
    Novembre 2006
    Messages
    79
    Détails du profil
    Informations personnelles :
    Âge : 47

    Informations forums :
    Inscription : Novembre 2006
    Messages : 79
    Par défaut Multithread en WPF
    Bonnour,

    J'essaye de faire du multithread en WPF avec un exemple tout simple et en utilisant l'objet Dispatcher :

    Lorsque l'utilisateur clique sur le bouton, le Dispatcher appelle la méthode BeginInvoke pour déclencher dans un thread séparé la méthode tacheLongue
    qui est une boucle For.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
     
    public partial class Window1 : Window
        {
     
          ....
     
            public delegate void NextPrimeDelegate();
     
     
     
            public void tacheLongue()
            {
                for (int i = 0; i < 100000; i++)
                {
                    label1.Content = label1.Content + i.ToString();
                }
     
            }
     
            private void button1_Click(object sender, RoutedEventArgs e)
            {
     
                Dispatcher.BeginInvoke(DispatcherPriority.Normal, new NextPrimeDelegate(tacheLongue));
     
     
            }
        }
    Le soucis c'est que l'application n'est pas multithread puisque tant que la boucle n'est pas fini l'interface utilisateur est bloquée, les autres contrôles de la fenêtre sont innaccessibles.

    Quelqu'un aurait une idée.

    Merci d'avance,

  2. #2
    Membre Expert
    Profil pro
    Inscrit en
    Juillet 2008
    Messages
    1 562
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Juillet 2008
    Messages : 1 562
    Par défaut
    ben normalement dans ton click tu devrais creer un thread qui va lancer la tache longue
    le disptacher sert dans ce cas, dans la tache longue, la boucle du thread pour renvoyer les informations vers l'interface principal
    en oubliant pas, que, si un click est refait alors que la tache longue n'est pas terminé on ne peut la relancer

  3. #3
    Rédacteur/Modérateur


    Homme Profil pro
    Développeur .NET
    Inscrit en
    Février 2004
    Messages
    19 875
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Développeur .NET
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Février 2004
    Messages : 19 875
    Par défaut
    Je vois 2 options :
    - utiliser un thread pour faire le traitement, et quand c'est fini, mettre à jour l'interface avec Dispatcher.Invoke
    - utiliser un BackgroundWorker et mettre à jour l'interface dans l'évènement RunWorkerCompleted, comme ça tu n'as plus besoin du Dispatcher

  4. #4
    Rédacteur
    Avatar de Thomas Lebrun
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    9 161
    Détails du profil
    Informations personnelles :
    Âge : 43
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 9 161
    Par défaut
    En effet, ton Dispatcher n'est pas là pour lancer des tâches longues, il permet de réavoir la main sur le thread de l'UI depuis un autre thread.

  5. #5
    Membre confirmé
    Inscrit en
    Novembre 2006
    Messages
    79
    Détails du profil
    Informations personnelles :
    Âge : 47

    Informations forums :
    Inscription : Novembre 2006
    Messages : 79
    Par défaut
    Citation Envoyé par Thomas Lebrun Voir le message
    En effet, ton Dispatcher n'est pas là pour lancer des tâches longues, il permet de réavoir la main sur le thread de l'UI depuis un autre thread.
    Merci les gars pour ces corrections, j'avais effectivement perdu de vu l'intérêt du Dispatcher. Erreur de conception.

    Je me mets au courant et je vous tiens au courant

  6. #6
    Membre confirmé
    Inscrit en
    Novembre 2006
    Messages
    79
    Détails du profil
    Informations personnelles :
    Âge : 47

    Informations forums :
    Inscription : Novembre 2006
    Messages : 79
    Par défaut
    Avec ce code ça semble mieux marcher.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
     
    private string texte = null;
     
            public delegate void DelegAffichageDecale();
     
            public void AffichageDecale()
            {
                label1.Content = texte;
            }
     
     
            public void tacheLongue()
            {
                for (int i = 0; i < 100000000; i++)
                {
                    texte = i.ToString();
     
                }
     
            }
     
            private void button1_Click(object sender, RoutedEventArgs e)
            {
                Thread th = new Thread(new ThreadStart(tacheLongue));
                th.Start();
                Dispatcher.BeginInvoke(DispatcherPriority.Normal, new DelegAffichageDecale(AffichageDecale));
            }
    Les autres contrôles de la page sont actifs pendant que le traitement long se déroule (méthode tacheLongue())

    Seul bémol, je ne comprends pas pourquoi l'affichage du contenu du label n'est pas modifié en temps réel.

  7. #7
    Rédacteur
    Avatar de Thomas Lebrun
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    9 161
    Détails du profil
    Informations personnelles :
    Âge : 43
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 9 161
    Par défaut
    Essaye:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    private string texte = null;
     
            public delegate void DelegAffichageDecale();
     
            public void AffichageDecale()
            {
                label1.Content = texte;
            }
     
     
            public void tacheLongue()
            {
                for (int i = 0; i < 100000000; i++)
                {
                    texte = i.ToString();
     
                    Dispatcher.BeginInvoke(DispatcherPriority.Normal, new DelegAffichageDecale(AffichageDecale));
     
     
                }
     
            }
     
            private void button1_Click(object sender, RoutedEventArgs e)
            {
                Thread th = new Thread(new ThreadStart(tacheLongue));
                th.Start();
            }

  8. #8
    Membre confirmé
    Profil pro
    Inscrit en
    Avril 2007
    Messages
    156
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2007
    Messages : 156
    Par défaut
    Cela peut être ennuyeux de créer un type de délégué pour chaque tâche qui peut être longue.
    On peut tout simplement passer à la classe ThreadPool une méthode anonyme.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    ThreadPool.QueueUserWorkItem(delegate
    {
                    MaMethodeQuiEstyLongueExécuter();
                    Dispatcher.BeginInvoke(DispatcherPriority.Normal, new Action(AffichageDecale), null);
    });
    Un tips à connaitre aussi le Dispatcher à une méthode caché de l'intellisense CheckAccess qui permet de savoir si on est sur le thread d'affichage.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
     
     
    void WriteError(Exception ex)
    {
        if(Dispatcher.CheckAccess())
           //OnAffiche
        else
           Dispacther.Invoke(DispatcherPriority.Normal, new Action<Exception>(WriteError), ex)
    }

  9. #9
    Rédacteur
    Avatar de Thomas Lebrun
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    9 161
    Détails du profil
    Informations personnelles :
    Âge : 43
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 9 161
    Par défaut
    Citation Envoyé par Archeone Voir le message
    Un tips à connaitre aussi le Dispatcher à une méthode caché de l'intellisense CheckAccess qui permet de savoir si on est sur le thread d'affichage.
    Une autre de ces méthodes "cachées" est la méthode VerifyAccess

    D'ailleurs, pour bien faire, il faudrait toujours appeller ces méthodes lorsques l'on développe des contrôles WPF mais bon, on le fait rarement

  10. #10
    Rédacteur/Modérateur


    Homme Profil pro
    Développeur .NET
    Inscrit en
    Février 2004
    Messages
    19 875
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Développeur .NET
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Février 2004
    Messages : 19 875
    Par défaut
    Citation Envoyé par Thomas Lebrun Voir le message
    Une autre de ces méthodes "cachées" est la méthode VerifyAccess
    C'est clair que ces méthodes sont utiles (enfin surtout CheckAccess), je ne comprends pas trop pourquoi elles sont cachées

  11. #11
    Membre confirmé
    Profil pro
    Inscrit en
    Avril 2007
    Messages
    156
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2007
    Messages : 156
    Par défaut
    VerifyAccess lance une exception si on est pas sur le bon thread.

  12. #12
    Rédacteur/Modérateur


    Homme Profil pro
    Développeur .NET
    Inscrit en
    Février 2004
    Messages
    19 875
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Développeur .NET
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Février 2004
    Messages : 19 875
    Par défaut
    Citation Envoyé par Archeone Voir le message
    VerifyAccess lance une exception si on est pas sur le bon thread.
    C'est pourquoi j'ai précisé "surtout CheckAccess"

  13. #13
    Membre confirmé
    Profil pro
    Inscrit en
    Avril 2007
    Messages
    156
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2007
    Messages : 156
    Par défaut
    Ouep, mais personne n'avait expliqué a quoi servait la seconde méthode c'est pour ça que je précise qu'elle fait pareil que CheckAccess mais au lieu de retourner un bool elle lève une exception lorsqu'on est sur un thread différent.

    Espérons que toutes nos précisions serviront a quelqu'un dans quelques mois quand un utilisateur retrouvera le thread!

  14. #14
    Membre éprouvé
    Profil pro
    Inscrit en
    Juillet 2009
    Messages
    159
    Détails du profil
    Informations personnelles :
    Âge : 47
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Juillet 2009
    Messages : 159
    Par défaut
    Hello,

    J'ajoute ma pierre à l'édifice. J'avais rencontré ces problèmes quand je me suis amusé à écrire un petit outil de parcours de l'AD en WPF. J'en parle dans mon article ici : http://msdn.microsoft.com/fr-fr/netf.../cc563816.aspx dans la section : "Rendre l’application multi-threadée". Le meilleur lien est celui là pour moi qui reste limpide : http://msdn.microsoft.com/fr-fr/library/ms741870.aspx

    Pour terminer, je vous conseille également l'excellent article de mon copain Eric Vernié sur le sujet : http://msdn.microsoft.com/fr-fr/netf.../cc895446.aspx

    Bye,

    David Rousset
    Microsoft France

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. [WPF][MultiThreading/Composants] Petit probleme de conception ?
    Par chico200987 dans le forum Windows Presentation Foundation
    Réponses: 5
    Dernier message: 15/08/2010, 19h02
  2. [WinAPI C++] MultiThreading et PostMessage
    Par Gruik dans le forum Windows
    Réponses: 7
    Dernier message: 29/03/2004, 15h58
  3. [WinAPI C++] MultiThreading?
    Par Gruik dans le forum Windows
    Réponses: 2
    Dernier message: 25/03/2004, 00h08
  4. [Win32]App multithread
    Par billyboy dans le forum Windows
    Réponses: 5
    Dernier message: 25/09/2003, 09h57
  5. Multithreading sous HP Ux 11
    Par pykoon dans le forum Autres éditeurs
    Réponses: 1
    Dernier message: 18/10/2002, 23h36

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo