Professional Documents
Culture Documents
Par SoftDeath
www.siteduzero.com
Sommaire
1/22
Sommaire
Sommaire ........................................................................................................................................... 1 Lire aussi ............................................................................................................................................ 0 Introduction aux sockets .................................................................................................................... 2
Rcuprer une adresse IP avec InetAddress ................................................................................................................... 3 Qu'est-ce qu'un socket ? ................................................................................................................................................... 4 change de message ....................................................................................................................................................... 7
Ct Serveur ............................................................................................................................................................................................................. 10 Ct Client ................................................................................................................................................................................................................. 10 Utilisation des threads ............................................................................................................................................................................................... 10
www.siteduzero.com
2/22
Par
SoftDeath
Mise jour : 21/09/2009 Difficult : Difficile 812 visites depuis 7 jours, class 131/779 Bienvenue dans mon tout premier mini-tutoriel consacr aux sockets en langage Java. On ne va pas tout apprendre sur les sockets mais tudier le plus important concernant ces classes du rpertoire java.net. Avant de vous lancer dans la lecture de ce tutoriel, les deux premires parties de cysboy sur le langage Java ainsi que la lecture de celui sur les threads et les flux d'entres et sorties (1/2) sont primordiales pour bien suivre le cours !
Je vous recommande galement de lire le chapitre de Dalshim Bien fermer ses threads en Java si vous voulez bien matriser la fermeture de vos threads. Les sockets servent communiquer entre deux htes appels Client / Serveur l'aide d'une adresse IP et d'un port que j'appelle prise ; ces sockets permettront de grer des flux entrant et sortant afin d'assurer une communication entre les deux (le client et le serveur), soit de manire fiable l'aide du protocole TCP/IP, soit non fiable mais plus rapide avec le protocole UDP. Nous allons tudier le premier mode, le mode TCP/IP V ce qu'on peut raliser l'aide des sockets : oici des jeux en ligne ; des systmes distribus ; des espaces messengers comme MSN Messenger, Yahoo Messenger, ; des applications comme BitComet permettant de grer les fichiers .torrent que vous connaissez ; et bien d'autres choses.
Les sockets sont utiliss dans plusieurs autres langages, tels que : le langage C : ( lien vers un tutoriel) ; le langage C++ : (lien vers un autre tutoriel) ; le langage PHP : (lien vers un troisime tutoriel) ; l'Action Script : (aller lire le tutoriel) ; le Erlang : (lien vers le tuto concern) ; et bien d'autres.
Lire les parties histoire et dfinitions de tutoriels ci-dessus ne vous fera pas de mal. Ne tardons pas et commenons.
Sommaire du tutoriel :
Rcuprer une adresse IP avec InetAddress Qu'est-ce qu'un socket ? change de message TP : un mini-chat entre le client et le serveur !
www.siteduzero.com
3/22
A prsent, voyons les mthodes applicables un objet de cette classe : getHostName() : elle retourne le nom de la machine dont l'adresse est stocke dans l'objet. getAddress() : elle retourne l'adresse IP stocke dans l'objet sous forme d'un tableau de 4 octets. toString() : elle retourne un String qui correspond au nom de la machine et son adresse.
Et pour terminer un petit exemple : Code : Java import java.net.InetAddress; import java.net.UnknownHostException; public class Adressage { public static void main(String[] zero) { InetAddress LocaleAdresse ; InetAddress ServeurAdresse; try { LocaleAdresse = InetAddress.getLocalHost(); System.out.println("L'adresse locale est : "+LocaleAdresse ); ServeurAdresse= InetAddress.getByName("www.siteduzero.com"); System.out.println("L'adresse du serveur du site du zro est : "+ServeurAdresse); } catch (UnknownHostException e) { } e.printStackTrace();
} }
Et le rsultat est : Code : Console L'adresse locale est : softdeath/239.254.78.177 L'adresse du serveur du site du zro est : www.siteduzero.com/80.248.219.123
www.siteduzero.com
4/22
Fastoche, hein ? La classe InetAdress peut lever une exception de type UnknownHostException, tchez de ne pas l'oublier !
Nous savons maintenant comment rcuprer l'adresse IP de notre machine ou d'une machine distante, que diriez-vous si l'on l'utilisait pour crer notre tout premier socket ? C'est parti
En outre, java.net comprend la classe ServerSocket, qui met en oeuvre une sorte de prise que les serveurs peuvent utiliser pour couter et accepter les connexions des clients. Ce qui nous donne : Code : Java ServerSocket socketserver = new ServerSocket(numero_port);
Ainsi on obtient un objet de la classe ServerSocket sur un port spcifique : si ce dernier est 0, le socket est cre sur n'importe quel port libre. Il existe deux autres constructeurs ; l'un a deux paramtres, le premier est bien sr le numro de port et le nombre total de connexion simultanes acceptes, voyez : Code : Java ServerSocket socketserver = new ServerSocket(numer_port,nbr_max);
www.siteduzero.com
5/22
Quant au client, celui-ci connat le nom de la machine sur laquelle le serveur est en excution et le numro de port sur lequel il coute. Le client va demander une connexion au serveur en s'identifiant avec son adresse IP ainsi que le numro de port qui lui est li. Pour cela Java.net fournit une classe Socket qui met en oeuvre un ct d'une connexion bidirectionnelle entre votre programme Java et un autre programme sur le rseau. La classe Socket se trouve au sommet d'une plate-forme de charge de mise en oeuvre en cachant les dtails d'un systme particulier de votre programme Java. En utilisant la Socket classe au lieu de compter sur du code natif, vos programmes Java peuvent communiquer sur le rseau en une plate-forme indpendante de la mode. La cration d'un socket pour le client ncessite un des constructeurs suivants : Code : Java Socket socket = new Socket(param1, param2)
Le premier paramtre correspond l'identit du client, il peut tre une chaine de caractre ou de type InetAddress, param2 correspond au numro de port sur lequel on souhaite se connecter sur le serveur. Il est possible galement de spcifier son adresse local comme troisime paramtre et le numro de port local : Code : Java Socket socket = new Socket(adresse_distante, port_distant, adresse_locale, port_locale)
Aprs tentative de connexion, si tout va bien, le serveur accepte la connexion du client, et reoit un nouveau socket qui est directement li au mme port local. Il a besoin d'une nouvelle prise de sorte qu'elle puisse continuer couter le socket d'origine pour les demandes de connexion, tout t'en satisfaisant les besoins du client connect. V comment accepter une connexion oici d'un client : Code : Java Socket socketduserveur = socketserver.accept();
Une fois le socket cr, l'attente de connexion provenant du client se fait l'aide de la mthode accept(). La mthode accept() reste bloquante tant qu'elle n'a pas dtect de connexion.
Afin d'viter une attente infinie, il est possible de spcifier un dlai maximal d'attente l'aide d'un mutateur setSoTimeout. Si un timeout est une valeur strictement positive, l'appel accept() va gnrer une attente maximale gale timeout. Si ce temps expire, une exception de type InterruptedIOException est leve sans toute fois que la socket de type ServerSocket ne soit invalide. La lecture de ce timeout se fait l'aide de l'accesseur getSoTimeout().
www.siteduzero.com
6/22
Ct client, si la connexion est accepte, une socket est cr et le client peut utiliser la socket pour communiquer avec le serveur. L'tablissement d'une connexion peut lever une exception de type IOException. On va essayer d'tablir une communication. V ous avez tout les lments porte de main, vous pouvez dsormais tablir une connexion, au boulot !
...Ce n'tait pas si difficile que a, voyons la correction Le client et le serveur peuvent prsent communiquer par l'criture ou la lecture de leurs prises.
Secret (cliquez pour afficher) Serveur.java Code : Java import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; public class Serveur { public static void main(String[] zero) { ServerSocket socketserver Socket socketduserveur ; try { socketserver = new ServerSocket(2009); socketduserveur = socketserver.accept(); System.out.println("Un zro s'est connect !"); socketserver.close(); socketduserveur.close(); }catch (IOException e) { e.printStackTrace(); } ;
} }
Client.java Code : Java import import import import java.io.IOException; java.net.InetAddress; java.net.Socket; java.net.UnknownHostException;
public class Client { public static void main(String[] zero) { Socket socket; try { socket = new Socket(InetAddress.getLocalHost(),2009); socket.close();
www.siteduzero.com
7/22
} }
V ous venez d'tablir votre premire connexion entre un serveur et un client recevoir des messages.
Attention : il ne faut jamais oublier de fermer le socket ! Sinon si vous utilisez le mme port dans un autre programme, ce (l'action de fermer une socket) ne sera plus possible car il (le port) sera occup !
change de message
Une fois la connexion tablie et les sockets en possession, il est possible de rcuprer le flux d'entre et de sortie de la connexion TCP vers le serveur. Il existe deux mthodes pour permettre la rcupration des flux : getInputStream() de la classe InputStream. Elle nous permet de grer les flux entrant ; getOutputStream() de la classe OuputStream. Elle nous permet de grer les flux sortant.
Ces deux mthodes nous permettent donc de grer les flux en entre et en sortie. En gnral le type d'entre et sortie utilis est BufferedReader et InputStreamReader pour la lecture, PrintWriter pour l'criture. Mais on peut utiliser tous les autres flux. En se basant sur l'annexe du tuto Java : BufferedReader : cette classe permet de lire des caractres partir d'un flux tamponn, afin de faire des lectures plus rapides ; InputStreamReader : convertit un flux binaire en flux de caractres : elle convertit un objet de type InputStream en objet de type Reader ; PrintWriter : la classe PrintWriter ajoute un flux la possibilit de faire des critures sous forme de texte des types primitifs Java, et des chanes de caractres.
www.siteduzero.com
8/22
Serveur.java Code : Java import import import import import import import import import java.io.BufferedReader; java.io.IOException; java.io.InputStreamReader; java.io.PrintWriter; java.net.InetAddress; java.net.ServerSocket; java.net.Socket; java.net.UnknownHostException; java.io.PrintWriter;
public class Serveur { public static void main(String[] zero) { ServerSocket socketserver Socket socketduserveur ; BufferedReader in; PrintWriter out; try { socketserver = new ServerSocket(2009); System.out.println("Le serveur est l'coute du port "+socketserver.getLocalPort()); socketduserveur = socketserver.accept(); System.out.println("Un zro s'est connect"); out = new PrintWriter(socketduserveur.getOutputStream()); out.println("Vous tes connect zro !"); out.flush(); socketduserveur.close(); socketserver.close(); }catch (IOException e) { ;
www.siteduzero.com
9/22
} }
Client.java Code : Java import import import import import import import java.io.BufferedReader; java.io.IOException; java.io.InputStreamReader; java.io.PrintWriter; java.net.InetAddress; java.net.Socket; java.net.UnknownHostException;
public class Client { public static void main(String[] zero) { Socket socket; BufferedReader in; PrintWriter out; try { socket = new Socket(InetAddress.getLocalHost(),2009); System.out.println("Demande de connexion"); in = new BufferedReader (new InputStreamReader (socket.getInputStream())); String message_distant = in.readLine(); System.out.println(message_distant); socket.close(); }catch (UnknownHostException e) { e.printStackTrace(); }catch (IOException e) { } e.printStackTrace();
} }
Rsultat chez le serveur : Code : Console Le serveur est l'coute du port 2009 Un zro s'est connect
www.siteduzero.com
10/22
Ct Serveur
Aprs tablissement de la connexion, le serveur obtient son socket qu'il utilise pour grer le flux sortant l'aide de socketduserveur.getOutputStream() ; ensuite, l'aide de la mthode println on envoie un message au client, on utilise flush pour vider le buffer tout simplement. Et pour finir on ferme la connexion.
Ct Client
Aprs avoir obtenu notre socket, on utilise socket.getInputStream() pour rcuprer le flux sortant. La mthode readLine() nous permet de lire une chane de caractres. Il existe plusieurs autres mthodes telles : readInt() permettant de lire un entier ; readDouble() permettant de lire un nombre de type double ;
Pour finir, on affiche le message reu et on ferme notre socket. a a l'air trs simple premire vue pour deux htes, mais si l'on veut que plus que deux puissent communiquer entre eux la fois, comment faire ? D'o la ncessit d'utiliser les Threads.
www.siteduzero.com
11/22
Serveur.java Code : Java import java.io.IOException; import java.net.*; public class Serveur { public static void main(String[] zero){ ServerSocket socket; try { socket = new ServerSocket(2009); Thread t = new Thread(new Accepter_clients(socket)); t.start(); System.out.println("Mes employeurs sont prts !"); } catch (IOException e) { } e.printStackTrace();
class Accepter_clients implements Runnable { private ServerSocket socketserver; private Socket socket; private int nbrclient = 1; public Accepter_clients(ServerSocket s){ socketserver = s; } public void run() { try { while(true){ socket = socketserver.accept(); // Un client se connecte on l'accepte System.out.println("Le client numro "+nbrclient+ " est connect !"); nbrclient++; socket.close(); } } catch (IOException e) { e.printStackTrace();
www.siteduzero.com
12/22
Client.java Code : Java import java.io.IOException; import java.net.*; public class Client { public static void main(String[] zero){ Socket socket; try { socket = new Socket("localhost",2009); socket.close(); } catch (IOException e) { } e.printStackTrace();
Rsulats : Code : Console Mes employeurs sont prts ! Le client numero 1 est connect ! Le client numero 2 est connect ! Le client numero 3 est connect !
Si vous m'avez bien suivis vous ne devriez pas avoir du mal comprendre ce code. Ds que quelqu'un tape la porte, on demande un employer de lui ouvrir la porte, celui-l va le saluer et lui dire bye bye avec socket.close(); s'il ne veut pas parler (communiquer) .
www.siteduzero.com
13/22
et du ct du client, tte--tte avec le serveur aprs connexion : Code : Console Demande de connexion Connexion tablie avec le serveur, authentification : Entrez votre login : chabanus Entrez votre mot de passe : chabanus Je suis connect Votre message : salut Le serveur vous dit : salut Votre message : a va ? Le serveur vous dit : bien et toi ?
Allez au boulot ! J'espre que vous avez au moins essay. Cette solution n'est ni unique ni la meilleure, elle est simple et minimale pour vous montrer comment les threads doivent tre grs. V oyons la correction : Ct Serveur Secret (cliquez pour afficher) Accepter_connexion.java Code : Java import java.io.*; import java.net.*; public class Accepter_connexion implements Runnable{ private ServerSocket socketserver = null; private Socket socket = null; public Thread t1; public Accepter_connexion(ServerSocket ss){ socketserver = ss; } public void run() { try { while(true){ socket = socketserver.accept(); System.out.println("Un zro veut se connecter ");
www.siteduzero.com
14/22
Authentification .java Code : Java import import import import java.net.*; java.util.NoSuchElementException; java.util.Scanner; java.io.*;
public class Authentification implements Runnable { private Socket socket; private PrintWriter out = null; private BufferedReader in = null; private String login = "zero", pass = public boolean authentifier = false; public Thread t2; public Authentification(Socket s){ socket = s; } public void run() { try { in = new BufferedReader(new InputStreamReader(socket.getInputStream())); out = new PrintWriter(socket.getOutputStream()); while(!authentifier){ out.println("Entrez votre login :"); out.flush(); login = in.readLine(); out.println("Entrez votre mot de passe :"); out.flush(); pass = in.readLine(); if(isValid(login, pass)){ out.println("connecte"); System.out.println(login +" vient de se connecter "); out.flush(); authentifier = true;
null;
} else {out.println("erreur"); out.flush();} } t2 = new Thread(new Chat_ClientServeur(socket,login)); t2.start(); } catch (IOException e) { } System.err.println(login+" ne rpond pas !");
www.siteduzero.com
15/22
private static boolean isValid(String login, String pass) { boolean connexion = false; try { Scanner sc = new Scanner(new File("zero.txt")); while(sc.hasNext()){ if(sc.nextLine().equals(login+" "+pass)){ connexion=true; break; } } } catch (FileNotFoundException e) { System.err.println("Le fichier n'existe pas !"); } return connexion; } }
Chat_ClientServeur.java Code : Java import import import import import java.io.BufferedReader; java.io.IOException; java.io.InputStreamReader; java.io.PrintWriter; java.net.Socket;
public class Chat_ClientServeur implements Runnable { private private private private private Socket socket = null; BufferedReader in = null; PrintWriter out = null; String login = "zero"; Thread t3, t4;
} public void run() { try { in = new BufferedReader(new InputStreamReader(socket.getInputStream())); out = new PrintWriter(socket.getOutputStream()); Thread t3 = new Thread(new Reception(in,login)); t3.start(); Thread t4 = new Thread(new Emission(out)); t4.start(); } catch (IOException e) { System.err.println(login +"s'est dconnect "); }
} }
www.siteduzero.com
16/22
Emission .java Code : Java import java.io.IOException; import java.io.PrintWriter; import java.util.Scanner; public class Emission implements Runnable { private PrintWriter out; private String message = null; private Scanner sc = null; public Emission(PrintWriter out) { this.out = out; } public void run() { sc = new Scanner(System.in); while(true){ System.out.println("Votre message :"); message = sc.nextLine(); out.println(message); out.flush(); }
Reception.java Code : Java import java.io.BufferedReader; import java.io.IOException; public class Reception implements Runnable { private BufferedReader in; private String message = null, login = null; public Reception(BufferedReader in, String login){ this.in = in; this.login = login;
public void run() { while(true){ try { message = in.readLine(); System.out.println(login+" : "+message); } catch (IOException e) { } e.printStackTrace();
www.siteduzero.com
17/22
Serveur.java Code : Java import java.io.*; import java.net.*; public class Serveur { public static ServerSocket ss = null; public static Thread t; public static void main(String[] args) { try { ss = new ServerSocket(2009); System.out.println("Le serveur est l'coute du port "+ss.getLocalPort()); t = new Thread(new Accepter_connexion(ss)); t.start(); } catch (IOException e) { System.err.println("Le port "+ss.getLocalPort()+" est dj utilis !"); } } }
Ct Client Secret (cliquez pour afficher) Chat_ClientServeur.java Code : Java import java.io.*; import java.net.*; import java.util.Scanner; public class Chat_ClientServeur implements Runnable { private private private private private Socket socket; PrintWriter out = null; BufferedReader in = null; Scanner sc; Thread t3, t4;
www.siteduzero.com
18/22
} }
Client.java Code : Java import java.io.*; import java.net.*; public class Client { public static Socket socket = null; public static Thread t1; public static void main(String[] args) { try { System.out.println("Demande de connexion"); socket = new Socket("127.0.0.1",2009); System.out.println("Connexion tablie avec le serveur, authentification :"); // Si le message s'affiche c'est que je suis connect t1 = new Thread(new Connexion(socket)); t1.start();
} catch (UnknownHostException e) { System.err.println("Impossible de se connecter l'adresse "+socket.getLocalAddress()); } catch (IOException e) { System.err.println("Aucun serveur l'coute du port "+socket.getLocalPort()); }
} }
www.siteduzero.com
19/22
public void run() { try { out = new PrintWriter(socket.getOutputStream()); in = new BufferedReader(new InputStreamReader(socket.getInputStream())); sc = new Scanner(System.in); while(!connect ){ System.out.println(in.readLine()); login = sc.nextLine(); out.println(login); out.flush(); System.out.println(in.readLine()); pass = sc.nextLine(); out.println(pass); out.flush(); if(in.readLine().equals("connecte")){ System.out.println("Je suis connect "); connect = true; } else { System.err.println("Vos informations sont incorrectes "); } } t2 = new Thread(new Chat_ClientServeur(socket)); t2.start(); } catch (IOException e) { } System.err.println("Le serveur ne rpond plus ");
} }
www.siteduzero.com
20/22
Reception.java Code : Java import java.io.BufferedReader; import java.io.IOException; public class Reception implements Runnable { private BufferedReader in; private String message = null; public Reception(BufferedReader in){ } this.in = in;
public void run() { while(true){ try { message = in.readLine(); System.out.println("Le serveur vous dit :" +message); } catch (IOException e) { } e.printStackTrace();
} }
www.siteduzero.com
21/22
Lancez serveur.java et ensuite client.java, n'oubliez pas de crer un fichier pour le serveur pour qu'il puisse y vrifier l'identit de ses clients, je l'ai nomm zero.txt : vous pouvez le nommer comme vous le souhaitez, c'est votre guise. Optimisation : Pour ne pas dire qu'il en existe l'infini, plusieurs autres possibilits s'offrent vous, vous pouvez : rendre l'application client / client. Le serveur authentifie les deux clients et ces deux-l pourront discuter ensemble sans que d'autres puissent y entrer : grosso modo raliser un salon , comme on l'appelle ; utiliser les bases de donnes pour la sauvegarde des messages et pour authentifier les clients ; raliser une interface graphique en swing ou en awt ;
Croyez-moi : lorsque vous matriserez les sockets en utilisant les threads, rien ne sera plus compliqu pour vous. continuation tous ; surtout n'hsitez pas l'amliorer et envoyez-moi un MP pour m'exposer votre travail.
Bonne
Q.C.M.
Quelle est la classe qui permet de crer un socket pour le serveur ?
SocketServer. ServeurSocket. SocketServeur. ServerSocket.
Qu'est-ce qui cloche dans ce code ? Code : Java try { ServerSocket ss= new ServerSocket(1026); ss.close(); Socket s = ss.accept(); } catch (IOException e) { }
On a oubli d'utiliser les threads. On doit d'abord instancier un objet Socket avant ServerSocket. On a coup l'coute du serveur avant d'accepter les clients. On devrait mettre 2009 la place de 1026. Le port 1026 est un port rserv.
Correction !
Statistiques de rponses au Q CM
Le mode UDP est plus rapide que le mode TCP mais c'est une communication non fiable, car on ne se proccupe pas du rsultat de la rception du rcepteur, il est donc moins utilis. Maintenant que vous savez manier les sockets, le mode UDP ne devrait pas vous poser de problme . Je vous remercie d'avoir lu le tutoriel, il est prsent termin. J'espre que vous avez bien suivi la prochaine pour un autre tutoriel client/serveur ! La technologie RMI (Remote Method Invocation) ! Qui vous permettra de
www.siteduzero.com
22/22
Partager
Ce tutoriel a t corrig par les zCorrecteurs.
www.siteduzero.com