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

Java Discussion :

récupération d'une référence de Thread


Sujet :

Java

  1. #1
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut récupération d'une référence de Thread
    Bonjour à tous,

    j'ai un probleme de conception, avant de me lancer dans le codage je veux être sur de faire le bon choix d'archi pour mon appli...

    Je vais tenter d'être clair.

    J'ai un Thread A qui fait un pooling d'une connexion sur port USB et qui attend des commandes à envoyer.

    Par contre ce Thread A est créé au démarrage de mon appli et cette même appli démarre une Servlet B qui écoute des commandes distantes.

    Donc mon problème est que ma Servlet B en recevant un message doit envoyer ce message à mon Thread A mais n'a pas de référence sur ce Thread...
    Donc pour ma Servlet B (ou même un autre Thread), comment récupérer la référence sur ce Thread A sans avoir le contexte principal ?

    J'avais pensé stocker la référence du Thread (Thread.getId()) dans un fichier et le récupérer depuis ma Servlet B en créant un objet du même type et lui renseigner la bonne référence de l'objet, mais je ne suis pas sur que l'on puisse faire ça.

    Pourriez vous m'éclairer sur les pratiques de récupération de référence de Thread ? je suis sur que ce problème est fréquent et j'ai toujours réussi à pallier à ce probleme via des comm par fichier mais je me rend compte que ce n'est pas forcément très fiable...

    Merci de votre aide.
    Tiamat.

  2. #2
    Nouveau membre du Club
    Profil pro
    Inscrit en
    Février 2008
    Messages
    7
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Février 2008
    Messages : 7
    Par défaut
    Je ne comprends pas bien ton besoin... tu veux faire communiquer 2 processus entre eux ? Pourquoi ne pas utiliser RMI ?

  3. #3
    Modérateur
    Avatar de Alkhan
    Homme Profil pro
    ingénieur full stack
    Inscrit en
    Octobre 2006
    Messages
    1 232
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : ingénieur full stack

    Informations forums :
    Inscription : Octobre 2006
    Messages : 1 232
    Par défaut
    bonjour,

    Si c'est ton appli qui lance ta thread A et ta servlet B alors pourquoi ne pas passer la référence de A lors de l'instanciation de B ?
    Il n'y a pas de problème, il n'y a que des solutions.
    Cependant, comme le disaient les shadoks, s'il n'y a pas de solution, c'est qu'il n'y a pas de problème.
    Si toutefois le problème persiste, la seule solution restante est de changer le périphérique qui se trouve entre la chaise et l'écran

    Mes Articles : Mon premier article est sur le language D
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java

  4. #4
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Le design pattern singleton !
    Ton Thread est créé dans une classe Pool. Tu fais ainsi :
    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
    public class Pool
    {
      private static Pool instance;
     
      private Pool()
      {
        ...
      }
     
      private Pool( blabla )
      {
        ...
      }
     
      public static Pool getInstance()
      {
        if( instance == null ) { instance = new Pool(); }
        return instance;
      }
    }
    Et dans ta Servlet :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Pool monPool = Pool.getInstance();
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  5. #5
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    Salut Dinobogan,

    Merci pour ta réponse.
    donc mettons le cas dans mon main, je crée une class MonThread qui va créer un Thread, et en ajoutant une méthode:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    public static MonThread getInstance()
      {
        if( instance == null ) { instance = new Pool(); }
        return instance;
      }

    De ce fait dans ma Servlet B, je peux faire:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    MonThread thread = MonThread.getInstance();
    et je récupèrerais l'instance du thread MonThread présent dans la JVM ?
    c'est bien ça ?

    désolé mais je ne connais pas trop les design pattern singleton...
    Merci,

    Tiamat.

  6. #6
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Oui, c'est exactement ça.

    Si tu veux empêcher les objets de conserver une instance de "MonThread", tu peux aussi passer uniquement par des méthodes static pour commander le thread, à toi de voir. Du coup, la méthode "getInstance" devient private et toutes les méthodes d'accès deviennent static :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    public static void envoiMessage( String s )
    {
      MonThread thread = getInstance();
      thread.traiteMessage( s );
    }
     
    ...
    A toi de voir selon tes besoins
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  7. #7
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    Merci beaucoup pour ta réponse !

    par contre j'ai une question qui me vient du coup... dans le cas ou j'ai plusieurs instances de MonThread (ce qui n'est pas le cas...), de ce fait la méthode getInstance() n'a pas la possibilité de venir récupérer le bon objet par cette méthode ?

    Je vais faire un essai comme ça et je reviens vers vous.
    Merci,

    Tiamat.

  8. #8
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Citation Envoyé par tiamat Voir le message
    Merci beaucoup pour ta réponse !

    par contre j'ai une question qui me vient du coup... dans le cas ou j'ai plusieurs instances de MonThread (ce qui n'est pas le cas...), de ce fait la méthode getInstance() n'a pas la possibilité de venir récupérer le bon objet par cette méthode ?

    Je vais faire un essai comme ça et je reviens vers vous.
    Merci,

    Tiamat.
    Jamais tu ne pourras avoir plus d'une instance de MonThread. La méthode "getInstance" te renverra toujours la même instance, quand bien même tu l'appellerais mille fois. De plus, tous les constructeurs sont private, donc il n'y a aucun autre moyen d'obtenir une instance que de passer par "getInstance".
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  9. #9
    Membre émérite
    Avatar de divxdede
    Profil pro
    Inscrit en
    Avril 2004
    Messages
    525
    Détails du profil
    Informations personnelles :
    Âge : 47
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Avril 2004
    Messages : 525
    Par défaut
    Il me semble qu'il s'agit plus d'un problème de synchronisation.

    Le Thread A devrait mettre à disposition un objet de synchronisationsur lequel il serait en atttente (wait)

    Les threads (de tes servlet) pourrait alors lancer un notifyAll() aprés avoir inscrit le message dans la liste des messages à traiter.

    Typiquement

    ThreadA

    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
           while( !demandeArret) {
                    Message message = null;
                    synchronized(demandes) {
                           while( demandes.isEmpty() ) {
                                try {
                                    demandes.wait();
                                }
                                catch(InterruptedException ie ) {
                                     /* do nothing */
                                }
                            }
                            message = demandes.pop(); // prends un message en le retirant de la liste des messages à traiter
                     }
                     if( message != null ) {
                           // traiter le message
                           // le traitement de ce message ne bloque pas "demandes"
                           // ce qui permet aux autres threads de continuer a inscrire
                           // demandes
                     }
                 }
    ThreadB
    /** Ce code devrait être directement écrit dans l'objet "demandes"
    mais comme je ne voulais pas l'écrire ici
    */
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
      private void sendMessage( Message mess ) {
                       synchronized( demandes ) { 
                            demandes.add( mess );
                            demandes.notifyAll(); // reveil le ThreadA si besoin
                       }
                 }
    De cette façon, ton ThreadA ne se reveillera QUE lorsque des demandes sont à traiter et tu n'as pas besoin de connaitre le ThreadA.
    Par contre la problèmatique se déplace puisque tes autres threads auront besoin de connaitre l'instance de l'objet permettant d'inscrire des demandes.

    Ce code n'est qu'un exemple, je ne sais pas ce que sont tes messages, et je n'ai pas écrit l'objet "demandes" qui n'est même pas typé dans l'exemple.
    J'ai écrit ce code directement sur ce formulaire donc des erreurs de typos peuvent s'être glissées.

  10. #10
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    bon j'ai testé avec la méthode getInstance dans mon ThreadA, qui est appellée par ma Servlet B, par contre je vérifie l'ID du Thread et je m'aperçois que les ID ne correspondent pas...
    le Thread A est créé avec un Thread.id de 64, et lorsque je fais un getInstance dans ma servlet, l'instance.id du ThreadA récupéré est 65...

    donc après je ne sais pas si il n'y a toujours qu'une seule instance du Thread A en mémoire ou si il en a créé deux...

  11. #11
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Comment la classe MonThread fabrique une nouvelle instance de Thread ?
    Si tu as moyen de poster le bout de code utilisé, ce sera plus facile.
    Un id de Thread identifie un thread. Deux numéros différents décrivent deux threads différents. Tu as coder le singleton de manière un peu bancale
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  12. #12
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    voici le bout de code:

    ThreadUSB:
    j'initie l'instance dans mon Thread
    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
    29
    30
    31
     
    private static USBThread instance;
    ...
    	public static USBThread getInstance()
    	  {
    	    if( instance == null ) { instance = new USBThread(); }
    	    return instance;
    	  }
     
    	public void run(){
    		instance = this;
    		cmd = null;
    		log.debug("Démarrage Thread USB "+Thread.currentThread().getId());
    		com = new ComUSB();
    		com.start();
     
    		while(fin != 1){
    			synchronized(this){
    			try{
    				this.wait();
    				log.debug("Traitement du message");
    				process();				
    				cmd = null;				
    			}catch(Exception e){
     
    			}
    			}
     
    		}
     
    	}
    et dans mon autre bout de code je fais:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    		USBThread usb = USBThread.getInstance();
    		synchronized(usb){
    		usb.sendCommand(cmd);
    		usb.notify();
    		}
    en faisant cela, l'instance récupérée par getInstance est différente de celle créée au début.

    Merci de ton aide.
    Tiamat.

  13. #13
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    à moins que le fait que je mette mon Thread en mode wait() empêche de récupérer sa référence ? (une question en passant...)

  14. #14
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    As-tu fais une erreur de frappe lorsque tu dis que ThreadUSB contient "getInstance" qui renvoie une instance de USBThread ?
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  15. #15
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Citation Envoyé par tiamat Voir le message
    à moins que le fait que je mette mon Thread en mode wait() empêche de récupérer sa référence ? (une question en passant...)
    Non, car la référence est obtenue via un autre thread. C'est le code dans la méthode "run" qui est mis en parallèle.
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  16. #16
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    non non c'est bien ce que j'ai codé.

    j'ai une classe ThreadUSB qui a les méthodes suivantes:

    sendCommand()
    fin()
    run()
    process()
    public static USBThread getInstance()

    par contre je crée l'objet instance comme ceci:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    private static USBThread instance;
    et je charge l'instance dans la méthode run() par:
    au démarrage du Thread pour être sur qu'il sera la seule référence par la suite.

    pourquoi ? ai je fait une erreur ?

  17. #17
    Membre émérite Avatar de Spoutnik
    Homme Profil pro
    Inscrit en
    Octobre 2003
    Messages
    672
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 45
    Localisation : France

    Informations forums :
    Inscription : Octobre 2003
    Messages : 672
    Par défaut
    J'ai pas lu en détail la discution, mais déjà une chose, la méthode de singleton est fausse.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    public class Pool
    {
      private static Pool instance;
     
      private Pool(){}
     
      public static Pool getInstance()
      {
        if( instance == null ) { instance = new Pool(); }
        return instance;
      }
    }
    le bloc getInstance n'est pas synchronisé et donc, il peut y avoir x objects au lieu d'un seul.

    Si un premier appel a passé le test "if (instance == null)", avant que l'instance de Pool soit créée et affecté à la variable "instance", il peut y avoir un deuxième appel qui va lui aussi passer au travers du test et donc provoquer une nouvelle instanciation de Pool.


    Il faut utilser :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    public class Pool
    {
      private static Pool instance;
      
      private Pool(){}
    
      public synchronized static Pool getInstance()
      {
        if( instance == null ) { instance = new Pool(); }
        return instance;
      }
    }
    Personnellement, je préfère ceci :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    public class Pool
    {
      private static Pool instance = new Pool();
     
      private Pool(){ }
     
      public static Pool getInstance()
      {
        return instance;
      }
    }
    [EDIT]
    Je viens de lire un peu plus ton problème.
    Si je comprend bien, tu as une servlet qui récupère des commandes à renvoyer sur le port USB?

    Si c'est bien ca , regarde du coté de la classe ConcurrentLinkedQueue qui te permet de mettre en place une architecture producteur/consommateur qui devrait te rendre service.

    edit2 :
    http://louis.cova.neuf.fr/blocs-notes/page14.html
    http://www.javalobby.org/java/forums/m91820807.html


    ++

  18. #18
    Modérateur
    Avatar de dinobogan
    Homme Profil pro
    ingénieur
    Inscrit en
    Juin 2007
    Messages
    4 073
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : ingénieur
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juin 2007
    Messages : 4 073
    Par défaut
    Bien vu pour le synchronized
    J'ai écrit en vite flash, mais ce n'est pas une excuse

    Sinon, qui fait l'appel à "start" pour démarrer le thread ?
    Ce n'est pas un singleton que tu as créé.

    Et du coup je ne comprends plus : si tu fais "instance=this", tu es dans la classe USBThread. Or tu me dis que la méthode "run" est dans ThreadUSB
    Y-aurait-il une relation d'héritage entre ThreadUSB et USBThread ?
    Tu devrais utiliser le singleton comme il faut, en cachant la classe USBThread à l'utilisateur.
    N'oubliez pas de consulter les FAQ Java et les cours et tutoriels Java
    Que la force de la puissance soit avec le courage de ta sagesse.

  19. #19
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    Salut Spoutnik,

    merci de ton post, j'avais effectivement lu sur un article qu'il y avait un risque d'accès conccurent si on ne synchronizait pas le getInstance, ce que j'ai fait, donc pas de soucis la dessus.

    après quelques tests, il semblerait que cela tourne correctement, mais un truc qui m'échappe c'est qu'en récupérant le Thread.getId() de chaque coté (Thread A d'un coté, et dans ma Servlet), je me rend compte que l'ID n'est pas le même...

    par contre ils ont l'air de bosser avec le meme objet...

    je vais regarder la classe ConcurrentLinkedQueue, meme si j'ai déjà établi mon protocole (basé sur XML), mais ça vaut quand meme la peine de jeter un coup d'oeil.

  20. #20
    Membre confirmé
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    139
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 139
    Par défaut
    oui pardon y a que USBThread...ThreadUsb je l'ai écrit un peu vite...

    voici le déroulement.
    un thread main crée le USBThread:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    USBThread usb = new USBThread();
    		usb.start();
    la méthode run() de mon USBThread initie l'instance:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    	public void run(){
    		instance = this;....
    la classe USBThread contient aussi un getInstance():
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    	public synchronized static USBThread getInstance()
    	  {
    	    if( instance == null ) { instance = new USBThread(); }
    	    return instance;
    	  }
    et ma servlet qui a besoin de récupérer la référence de USBThread fait ceci:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    USBThread usb = USBThread.getInstance();
    		synchronized(usb){
    		usb.sendCommand(cmd);
    		usb.notify();
    		}
    j'espère que c'est un peu plus clair...
    A+

+ Répondre à la discussion
Cette discussion est résolue.
Page 1 sur 2 12 DernièreDernière

Discussions similaires

  1. Récupération d'une exception dérivée dans un thread
    Par Baccs dans le forum Threads & Processus
    Réponses: 4
    Dernier message: 22/08/2008, 15h55
  2. [1.1] passer une référence dans un thread
    Par melleb dans le forum C#
    Réponses: 4
    Dernier message: 26/05/2008, 11h17
  3. Réponses: 4
    Dernier message: 11/06/2004, 16h27
  4. [C#] Récupération d'une image depuis une table SQL Server
    Par borgfabr dans le forum Accès aux données
    Réponses: 10
    Dernier message: 08/04/2004, 13h20
  5. Oracle Designer: récupération d'une vue dans la base
    Par BILLYPATOU dans le forum Designer
    Réponses: 2
    Dernier message: 19/03/2004, 11h08

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