[JAVA] [API Javax Comm] Lire Port COM

Fermé
goddet Messages postés 6 Date d'inscription lundi 12 mars 2007 Statut Membre Dernière intervention 5 novembre 2007 - 24 avril 2007 à 09:27
 amoun - 2 mars 2009 à 20:13
Bonjour à tous!
J'utilise dans un programme java l'API javax.comm pour lire des infos venant d'un recepteur gps USB (détecté sur un port série virtuel).
Etant donné que le numéro du port COM attribué au recepteur GPS peut changer et que l'utilisateur du programme ne doit pas s'occuper de ça, je scanne tous les ports série et test si il y a des datas format NMEA (trames du recepteur GPS) qui arrivent du port COM en question. (Je n'ai pas trouver mieux!)
Donc j'essaye de faire une lecture du port série sur évènements comme dans ce tutoriel :
https://christophej.developpez.com/tutoriel/java/javacomm/#L2.3.3

J'ai juste un petit souci maintenant : j'arrive bien a trouver mon recepteur GPS mais mon prog plante une fois les trames lues! Je crois que le prog reste bloqué dans la structure switch qui gère les évènements du port série!
import javax.comm.*;
import java.util.*;
import java.math.*;
import java.net.*;
import com.sun.comm.Win32Driver;
import java.io.*;

public class testCOM implements SerialPortEventListener {
	
	private String portCOM;
    private CommPortIdentifier portID = null; //identifiant du port
    private SerialPort serialPort; //le port série
    private BufferedReader fluxLecture; //flux de lecture du port

	
	/*
	 * Methode qui initialise le port série en evenementiel
	 */
	public void ModeEvenement(String portCOM) {
		//récupération de l'identifiant du port
		try {
			portID = CommPortIdentifier.getPortIdentifier(portCOM);
		} catch (NoSuchPortException e) {
		}
		
		//ouverture du port
		try {
			serialPort = (SerialPort) portID.open("ModeEvenement", 2000);
		} catch (PortInUseException e) {
		}
		
		//récupération du flux
		try {
			fluxLecture =
				new BufferedReader(
					new InputStreamReader(serialPort.getInputStream()));
		} catch (IOException e) {}
		
		//ajout du listener
		try {
			serialPort.addEventListener(this);
		} catch (TooManyListenersException e) {
		}
		//paramétrage du port
		serialPort.notifyOnDataAvailable(true);
		try {
			serialPort.setSerialPortParams(
				4800,
				SerialPort.DATABITS_8,
				SerialPort.STOPBITS_1,
				SerialPort.PARITY_NONE);
		} catch (UnsupportedCommOperationException e) {}
		System.out.println("port ouvert, attente de lecture");
	}	
    
    /*
     * Methode qui effectue la lecture de 7 trames sur le port série
     * Une fois la lecture faite on ferme le flux de lecture et le port COM
     */
    public void ReadSerialPort(){
    	int i=7;
    	String reponse = new String();
		
		try {
			System.out.println("i="+i);
			while(i!=0){
				System.out.println("On lit sur le port COM\n");
				reponse = (String) fluxLecture.readLine();
				System.out.println(reponse);
				i--;
				System.out.println("i="+i);
			}
		} catch (IOException e) {}
		//fermeture du flux de lecture
		try {
			fluxLecture.close();
		} catch (IOException e) {}
		//fermeture du port COM
		serialPort.close();
	}
	
	public void serialEvent(SerialPortEvent event) {
		//gestion des événements sur le port :
		//on ne fait rien sauf quand les données sont disponibles
				switch (event.getEventType()) {
			case SerialPortEvent.DATA_AVAILABLE :
				this.ReadSerialPort();//si data dispo on lance la lecture
				break;
			default:
				break;//on ne fait rien pour les autres evenements
		}
	}
    
    /*
     * Methode qui scanne tous les ports COM et test si des données viennent du port COm scanné
     */	
	public void listPort(){
		Enumeration listePorts = CommPortIdentifier.getPortIdentifiers();
		int typePort;
		String GPSPortCOM;
		
		while (listePorts.hasMoreElements()){
			portID = (CommPortIdentifier) (CommPortIdentifier) listePorts.nextElement();
			if(portID.getPortType()==CommPortIdentifier.PORT_SERIAL){    
  				System.out.println("Nom du PORT :"+portID.getName());
  				System.out.println("User :"+portID.getCurrentOwner());
  				System.out.println("Use ? :"+portID.isCurrentlyOwned());
  				System.out.println("Type du PORT :"+portID.getPortType());
  				// On lance la gestion des évènements sur portID
				this.ModeEvenement(portID.getName());
			}
		}
	}

	public static void main(String[] args) {
		//initialisation du driver
		Win32Driver w32Driver = new Win32Driver();
		w32Driver.initialize();
		
		testCOM test = new testCOM();
		test.listPort();
	}
}

Je ne sais pas si j'ai été assez clair! Mais si quelqu'un a une idée je suis preneur!!!
Merci d'avance.
A voir également:

2 réponses

Salut!

Il me semble que ton approche n'est pas la bonne, en fait ça m'étonne déjà que t'arrives à récupérer des trames comme ça.

Si j'ai bien suivi ça donne ça :
- tu crées ton objet testCOM
- tu lances la méthode ModeEvenement() de cet objet qui initialise le port et lui ajoute un listener, cette méthode se termine avec succès et après tu comptes sur l'objet lui-même pour gérer la suite par l'intermédiaire de sa méthode serialEvent.
- si tu reçois des données sur le port, serialEvent appelle la méthode ReadSerialPort() qui lit 7 trames puis ferme le port

En partant de là, je vois plusieurs soucis :

T'as aucun contrôle sur ce qui se passe, ton objet testCOM continue à recevoir des event qui t'intéressent ou pas, et ça tant qu'il n'est pas détruit, mais en plus t'as fermé le port entre temps...
Ensuite, cette façon de lire sept trames à la suite, c'est pas l'idéal, pourquoi tu continues pas d'utiliser serialEvent pour gérer trame par trame?

Bref, moi ce que j'en dis c'est :
- tu devrais vraiment, vraiment commencer par faire un thread de cette classe (comme préconisé à la fin du tuto dont tu parles, pour la gestion du lecteur de codes barres)!
En fait reprends carrément le code du tuto tel quel et changes juste la condition d'arrêt.
tu rajoutes comme variables globales à ta classe :
private int nbreTramesLues = 0;
private final int NBRE_TRAMES_A_LIRE = 7;

et au lieu de mettre dans le run(), while(running) comme dans le tuto, tu mets
while(this.nbreTramesLues < this.NBRE_TRAMES_A_LIRE)


- et puis pour les trames, lis trame par trame et pas les sept d'un coup:

public void serialEvent(SerialPortEvent event) {
		//gestion des événements sur le port :
		//on ne fait rien sauf quand les données sont disponibles
				switch (event.getEventType()) {
			case SerialPortEvent.DATA_AVAILABLE :
				try{
				 
				  System.out.println("On lit sur le port COM\n");
				  reponse = (String) fluxLecture.readLine();
				  System.out.println(reponse);
				  this.nbreTramesLues++;
				  
        }catch(IOException e){
        }
				break;
			default:
				break;//on ne fait rien pour les autres evenements
		}
	}


Dans le main tu lances juste ton thread en appelant la méthode start() (attention, pas run()directement!), si tout vas bien, le thread s'arrête de lui même une fois les sept trames reçues et le port
est fermé, plus de problème!
Enfin ça c'est si tout va bien, même comme ça si t'as pas tes sept trames, ton thread va rester bloqué indéfiniment dans le while, il faudra arranger ça, mais c'est déjà un début.

Voilà, après j'ai pas bien compris, tu fais ça pour chaque port série? Qu'est ce que tu fais si tu reçois rien sur un port? T'attends indéfiniment qu'il y ait des données qui arrivent?

Un truc encore, les méthodes commencent par une minuscule en java et les classes par une majuscule, là t'as fait exactement l'inverse! c'est plus facile de se repérer si tu suis ces conventions d'écriture.

Voilà ce n'est que mon avis sur la question, j'éspère que ça pourra t'aider un peu!

bye!
2
je presque le meme problème je veux envoyer des commandes AT vers mon prtable samsung connecté via usb le prog affiche deux com 4 et 5 plus que 3 qui est (com3 c'est le com de mon modem pc)
lorsque j'envois ces commandes rien ne se passe (pas de résultat)
mais lorsque je ferme ou je j'ouvre le clavier des caractéres bizaares et des carreaux s'affiches et mot locked ou un locked détirées. je croix que mon portable ne recoit pas ces commandes .
si vous avez recontre ce prob contacte moi sur msn [supprimé par la modération] j e vraiment besoin de ton aide,je suis mnt de réaliser ce truc com mon pfe et il y on ad'autres proble......

voici ce code et j'attends vos idées sur mon hotmail :


/*Most projects that deal with hardware and devices, needs to communicate with them using the COM port of the PC or Server. For example if there is a modem that is connected to a server via its COM port and the Java program has to read the output of the modem then the Java program has to read the COM port for any incoming data.

This sample Java program can be used to Read from a COM port for incoming data and process it. Note that you will need to change the Port number to COM1 or COM2 or any other ports as required.

Also if you are using unix based machines then you will have to uncomment the /dev/term/a instead of COM.*/

import java.io.*;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.IOException;
import java.util.*;
import javax.comm.*;
import com.sun.comm.Win32Driver;

public class SimpleRead implements Runnable, SerialPortEventListener {
    static CommPortIdentifier portId;
    static Enumeration portList;

    static InputStream inputStream;
    static SerialPort serialPort;
    static Thread readThread;
    static OutputStream outputStream;
    static SerialPortEvent event;
    private static BufferedReader in;
    
    public SimpleRead() {
        try {
            serialPort = (SerialPort) portId.open("Envoi", 20000);
            }
        catch (PortInUseException e) {System.out.println(e);}

  /*  try {
            serialPort.addEventListener(this);
        }
       catch (TooManyListenersException e) {System.out.println(e);}*/

              serialPort.notifyOnDataAvailable(true);

        try {
                serialPort.setSerialPortParams(9600,
                SerialPort.DATABITS_8,
                SerialPort.STOPBITS_1,
                SerialPort.PARITY_NONE);
            }
        catch (UnsupportedCommOperationException e) {System.out.println(e);}

        readThread = new Thread(this);
        readThread.start();
    }

    public static void EcritureSurRS232(String data ) {
      try
      {
        outputStream = serialPort.getOutputStream();
        outputStream.write(data.getBytes());
      }
      catch (IOException e)
      {
       System.out.println("erreur dans l'écriture sur le port série\n" +
                                        e.getMessage());
      }
    }

    public static void lecturePortRS232() throws IOException{
        //byte[] readBuffer = new byte[2];
   char[] charbuffer=new char[100];
             //inputStream= serialPort.getInputStream();

               try {
                    //while (inputStream.available() > 0) {
                    //int numBytes = inputStream.read(readBuffer);
                   //}
                    //System.out.print(new String(readBuffer));
                          in=new BufferedReader(new InputStreamReader(serialPort.getInputStream()));
                         String codeBarre = new String();
                        try {
                               //lecture du buffer et affichage
                               codeBarre = (String) in.readLine();
                             System.out.println(codeBarre);
                            } catch (IOException e) {}
                  //in.close();
               }
               catch (IOException ex)
                   {
                   System.out.println("erreur dans la lecture du port\n"+ex.getMessage());
                   }

    }

    public void run() {
        try {
            Thread.sleep(20000);
        } catch (InterruptedException e) {System.out.println(e);}
    }

 public void serialEvent(SerialPortEvent event) {
      switch(event.getEventType()) {
        case SerialPortEvent.BI:
        case SerialPortEvent.OE:
        case SerialPortEvent.FE:
        case SerialPortEvent.PE:
        case SerialPortEvent.CD:
        case SerialPortEvent.CTS:
        case SerialPortEvent.DSR:
        case SerialPortEvent.RI:
        case SerialPortEvent.OUTPUT_BUFFER_EMPTY:
            break;
        case SerialPortEvent.DATA_AVAILABLE:
          byte[] readBuffer = new byte[20];

            try {
                while (inputStream.available() > 0) {
                    int numBytes = inputStream.read(readBuffer);
                }
                System.out.print(new String(readBuffer));
            } catch (IOException e) {System.out.println(e);}
            break;
        }
    }
        public static void main(String[] args) throws IOException {
        portList = CommPortIdentifier.getPortIdentifiers();

        while (portList.hasMoreElements()) {
            portId = (CommPortIdentifier) portList.nextElement();
            if (portId.getPortType() == CommPortIdentifier.PORT_SERIAL) {
                 if (portId.getName().equals("COM4")) 
                 {
                    SimpleRead reader = new SimpleRead();
                    System.out.println("saisir la commande :") ;
                    BufferedReader stdin =new BufferedReader(new InputStreamReader(System.in));
                    String data=stdin.readLine();
                    EcritureSurRS232(data);
                    lecturePortRS232();
                }
            }
        }
    }
}
0
je m'adresse en fait a goddet,je suis entrain de developper une application qui permet la navigation gps sur un pocket pc,comme première étape j'ai besoin de capter les trames nmea provenant du gps sur un port série,j'ai essayé de faire sa avec l'hyperterminal,et je vois que votre code va bien avec mon application?
est ce que vous avez trouvé solution à votre problème?
pouvez vous m'aidez?
1