POO Lab Lucrarea 5
De la WikiLabs
Noțiuni și cunoștințe necesare
- Paradigma Obiect-Orientare; Clase și obiecte
- Noțiuni despre Java
- Realizarea și execuția unui program Java
- Sintaxa limbajului Java; Structura lexicală a unui program
- Convenții de programare
- Noțiuni avansate de programare obiect-orientată
- Java Application Programming Interface (API)
- Stream-uri de Input/Output
- Tratarea excepțiilor
- Serializarea obiectelor
- Socket-uri de rețea
- Programare concurentă - fire de execuție (Threads)
Cerințe
- Modificați clasa
seriaf.poo.client.ClientPeer
scrisă la tema anterioară, transformând-o într-un fir de execuție care în paralel cu firul principal de execuție, să citească obiecte de tipseriaf.poo.structs.Message
dinspre server și să le afișeze pe ecran. Acest thread, odată instanțiat, trebuie pornit de către clasaseriaf.poo.client.TextClient
. - Modificați clasa
seriaf.poo.server.ServerPeer
scrisă la tema anterioară, transformând-o într-un fir de execuție. Acest fir nou de execuție trebuie să citească obiectele de tipseriaf.poo.structs.Message
șiseriaf.poo.structs.PrivateMessage
de la clientul asociat și să le distribuie corect la ceilalți clienți. - Adăugați clasei
seriaf.poo.structs.Message
o metodă getter publică pentru expeditor, numităString getSender()
. - Pentru a putea retransmite mesajele celorlalți clienți, clasa
seriaf.poo.server.ServerPeer
trebuie să aibă o metodă publicăvoid sendMessage(Message)
care să serializeze obiectul primit ca argument către clientul conectat. - Pentru a putea folosi clasa
seriaf.poo.server.Server
să trimită mesajele clienților conectați, clasaseriaf.poo.server.ServerPeer
trebuie să aibă o referință la clasaseriaf.poo.server.Server
. Astfel, modificați constructorul claseiseriaf.poo.server.ServerPeer
astfel încât să aibă semnăturaServerPeer(Server, Socket)
. - Pentru a putea trimite corect mesajele private, clasa
seriaf.poo.server.ServerPeer
trebuie să aibă o metodă publică numităString getUsername()
care să întoarcă numele client-ului conectat la acel ServerPeer. Numele este stocat în clasă într-un câmp privat, care se actualizează la fiecare mesaj primit, extrăgând din mesaj numele expeditorului. - Modificați clasa
seriaf.poo.server.Server
scrisă la tema anterioară, astfel încât la conectarea unui client, aceasta să creeze un nouseriaf.poo.server.ServerPeer
pe care să-l pornească ca Thread, apoi să revină în metodaServerSocket.accept()
, așteptând o nouă conexiune. - Clasa
seriaf.poo.server.Server
trebuie acum să fie instanțiabilă. Adăugați un constructor care să ia ca argument port-ul TCP pe care server-ul sa asculte și numărul maxim de clienți posibili. În acest constructor trebuie instanțiat obiectul de tipServerSocket
. - Bucla cu apelul metodei accept din
seriaf.poo.server.Server
trebuie acum să fie mutată într-o metodă separată a claseiseriaf.poo.server.Server
numităvoid listen()
. - În metoda main din
seriaf.poo.server.Server
trebuie să rămână o instanțiere de obiectseriaf.poo.server.config.ServerConfig
, o instanțiere de obiectseriaf.poo.server.Server
și un apel de metodă listen(), cu blocurile try-catch necesare. - Clasa
seriaf.poo.server.Server
trebuie să mai conțină două metode,void dispatch(Message)
șivoid removeClient(ServerPeer)
. Prima este apelată de către obiectele de tipseriaf.poo.server.ServerPeer
cu mesajele primite de la clienți, și are rolul de a le trimite mai departe, către ceilalți clienți. A doua metodă este apelată când un client de deconectează (voit sau datorită unei erori), pentru a fi elimiat din listă. - Server-ul nu trebuie să accepte o conexiune dacă numărul de clienți conectați este deja egal cu proprietatea MAX_CLIENTS citită din fișierul server.conf de către
seriaf.poo.server.config.ServerConfig
. - Atenție care metode trebuie să fie sincronizate (pentru a trece testele, nu folosiți blocuri
synchronzied
, ci declarați toată metoda sincronizată). - Administrați excepțiile astfel încât dacă apare o eroare de conexiune, server-ul să fie robust și să continue să servească corect ceilalți clienți conectați și să poate accepta un client nou în locul clientului pierdut.
Note:
- Obiectele de tip
seriaf.poo.structs.Message
se trimit tuturor clienților conectați. Obiectele de tipseriaf.poo.structs.PrivateMessage
se trimit doar către expeditor și către destinatar. - Clasa Server ar trebui să păstreze o listă cu toți clienții conectați (cu obiectele de tip ServerPeer). Pentru acest lucru vă puteți folosi de clasa java.util.ArrayList.
- Această listă trebuie să fie menținută în sensul că noii clienți trebuie adăugați, iar clienții deconectați trebuie eliminați.
- Pentru a verifica dacă un obiect e instanță a unei clase anume, puteți folosi operatorul instanceof:
Object o = ois.readOject();
if(o instanceof String){
String s = (String)o;
System.out.print(s);
}else{
System.out.print("The object is not a String!");
}