PDA

View Full Version : [JAVA] input e output sulla stessa Socket


m0linas
17-11-2010, 11:57
Ciao a tutti, devo farvi una domanda riguardo alle Socket, perchè nonostante ci abbia provato e riprovato veramente non riesco a capire....
So che si può comunicare in entrambe le direzioni sulla stessa socket, ma nel mio programma non riesco in nessun modo a farlo....bisogna che le letture/scritture avvengano all'interno dello stesso ciclo (:fagiano:) ?
Vi spiego in 2 parole...ho creato un programma molto rozzo di file sharing p2p per un esame all'università, e funziona. Il problema è che per aggiornare le liste dei file (client invia la lista dei files locali da condividere, e il server gli rimanda indietro la lista completa dei files disponibili) mi tocca usare 2 socket diverse, una per l'invio e una per la ricezione.
La cosa è decisamente brutta, e vorrei che tutto avvenisse sulla stessa socket, dato che da quanto ho capito la cosa si può fare....
In pratica il client manda la lista sulla socket, il server legge dalla socket la lista, la appende alla lista che già possiede, e rimanda quest'ultima (tramite la stessa socket) al client, che la legge e me la visualizza in una JList.
Il problema è la parte in neretto, tutto il resto funziona tranquillamente....

Questo è il metodo lato client:

public void inviaLista()
{
Socket s = null;
try {
s = new Socket();
s.connect(new InetSocketAddress(ip, port));
s.setSoTimeout(timeout);
if (!listaLocali.isEmpty()) listaLocali.clear(); // elimino la vecchia lista e la creo nuova
salvaLista(new File(".")); // verranno condivisi solamente i file nella cartella corrente

BufferedOutputStream bout = new BufferedOutputStream(s.getOutputStream());
ObjectOutputStream out = new ObjectOutputStream(bout);
for (int i=0;i<listaLocali.size();i++)
{
out.writeObject(listaLocali.get(i));
}

// prende indietro la lista
ObjectInputStream inLista = new ObjectInputStream(new BufferedInputStream(s.getInputStream()));
FileProva infl = (FileProva)inLista.readObject();
try
{
while (infl != null)
{
synchronized(listaFiles)
{
listaFiles.add(infl);
}
infl = (FileProva)inLista.readObject();
}
}catch (EOFException eof) {}
synchronized(listaFiles)
{
for (int i=0;i<listaFiles.size();i++)
{
listmodel.addElement(listaFiles.get(i));
}
}
lista.setModel(listmodel);
lista.setSelectionMode(ListSelectionModel.SINGLE_INTERVAL_SELECTION);
lista.setLayoutOrientation(JList.VERTICAL);
lista.setVisibleRowCount(-1);
inLista.close();
out.flush();
bout.flush();
out.close();
s.close();
} catch (IOException ee) {System.out.println("Errore del client!" ); ee.printStackTrace();}
catch (ClassNotFoundException cnfe) {cnfe.printStackTrace();}
}

Questo è il thread sul server col quale il metodo precedente comunica:

public void run()
{
try
{
BufferedInputStream bin = new BufferedInputStream(s.getInputStream());
ObjectInputStream in = new ObjectInputStream(bin);
InetAddress client = s.getInetAddress();
synchronized(ServerProva.filesArray)
{
for (int pd=0;pd<ServerProva.filesArray.size();pd++)
{
if (ServerProva.filesArray.get(pd).getOwner().equals(client))
{
ServerProva.filesArray.remove(pd);
pd--;
}
}
}
FileProva inFile = (FileProva)in.readObject();
try
{
while (inFile != null)
{
synchronized(ServerProva.filesArray)
{
ServerProva.filesArray.add(inFile);
inFile = (FileProva)in.readObject();
}
}
}catch (EOFException eof) {}


// Invia al client la lista completa dei files disponibili
// sulla rete

BufferedOutputStream bout = new BufferedOutputStream(s.getOutputStream());
ObjectOutputStream outLista = new ObjectOutputStream(bout);
synchronized(ServerProva.filesArray)
{
for (int i=0;i<ServerProva.filesArray.size();i++)
{
FileProva f = ServerProva.filesArray.get(i);
outLista.writeObject(f);
}
}
in.close();
bout.flush();
outLista.flush();
outLista.close();
s.close();

} catch (InterruptedIOException e) {System.out.println("Timeout in ServerThread");e.printStackTrace();}
catch (IOException ee) {System.out.println("Errore di IO in ServerThread");ee.printStackTrace();}
catch (ClassNotFoundException cnfe) {cnfe.printStackTrace();}
}

E questo è l'errore che mi becco:

Errore del client!
java.net.SocketTimeoutException: Read timed out
at java.net.SocketInputStream.socketRead0(Native Method)
at java.net.SocketInputStream.read(SocketInputStream.java:129)
at java.io.BufferedInputStream.fill(BufferedInputStream.java:218)
at java.io.BufferedInputStream.read1(BufferedInputStream.java:258)
at java.io.BufferedInputStream.read(BufferedInputStream.java:317)
at java.io.ObjectInputStream$PeekInputStream.read(ObjectInputStream.java:2266)
at java.io.ObjectInputStream$PeekInputStream.readFully(ObjectInputStream.java:2279)
at java.io.ObjectInputStream$BlockDataInputStream.readShort(ObjectInputStream.java:2750)
at java.io.ObjectInputStream.readStreamHeader(ObjectInputStream.java:780)
at java.io.ObjectInputStream.<init>(ObjectInputStream.java:280)
at ClientProva.inviaLista(ClientProva.java:128)
at ClientProva.aggiorna(ClientProva.java:167)
at ClientProva.main(ClientProva.java:202)

Nel lato client, se chiudo lo stream in uscita appena ho scritto sulla socket, quando vado a fare s.getInputStream mi dice

java.net.SocketException: Socket is closed
at java.net.Socket.getInputStream(Socket.java:788)
at ClientProva.inviaLista(ClientProva.java:131)
at ClientProva.aggiorna(ClientProva.java:168)
at ClientProva.main(ClientProva.java:203)
Errore di IO in ServerThread
java.net.SocketException: Socket closed
at java.net.SocketOutputStream.socketWrite(SocketOutputStream.java:99)
at java.net.SocketOutputStream.write(SocketOutputStream.java:136)
at java.io.BufferedOutputStream.flushBuffer(BufferedOutputStream.java:65)
at java.io.BufferedOutputStream.flush(BufferedOutputStream.java:123)
at ServerThread.run(ServerThread.java:75)

In pratica se chiudo subito uno dei 2 stream dello stesso processo, l'altro vede chiusa proprio la socket, se non li chiudo si pianta tutto e va in timeout....

É una settimana che il programma sarebbe finito, tranne per questa cagata qui....sarò infinitamente grato a chi mi risolve l'arcano!

banryu79
17-11-2010, 13:24
Credo che il problema sia in 'inviaLista', qui:

...
// prende indietro la lista
ObjectInputStream inLista = new ObjectInputStream(new BufferedInputStream(s.getInputStream()));
FileProva infl = (FileProva)inLista.readObject();
...

L'operazione di lettura dall'inpuStream del Socket ti va in timeout (come testimoniato dall'eccezione) perchè probabilmente dall'altra parte il server deve ancora inviare le informazioni che il tuo client si apetta di ricevere e l'inpuStream è vuoto.

m0linas
17-11-2010, 16:32
Credo che il problema sia in 'inviaLista', qui:

...
// prende indietro la lista
ObjectInputStream inLista = new ObjectInputStream(new BufferedInputStream(s.getInputStream()));
FileProva infl = (FileProva)inLista.readObject();
...

L'operazione di lettura dall'inpuStream del Socket ti va in timeout (come testimoniato dall'eccezione) perchè probabilmente dall'altra parte il server deve ancora inviare le informazioni che il tuo client si apetta di ricevere e l'inpuStream è vuoto.

Ciao, grazie per avermi risposto, come sempre. Molto gentile.
Allora, non ci pensavo che incollando il codice qua non mi riportava i numeri di riga del compilatore....la riga in cui da il timeout, è quella in cui istanzio l'inputstream, non quella dove faccio readObject().


...
// prende indietro la lista
ObjectInputStream inLista = new ObjectInputStream(new BufferedInputStream(s.getInputStream()));
FileProva infl = (FileProva)inLista.readObject();
...


Fra l'altro ho infilato Thread.sleep(3000) in inviaLista() fra l'invio sull'output e la ricezione dal server (il timeout della socket è 5000...), ma resta comunque 5 secondi bloccato, e poi mi da ancora read timed out........qualche idea?

banryu79
18-11-2010, 09:43
...
Allora, non ci pensavo che incollando il codice qua non mi riportava i numeri di riga del compilatore....la riga in cui da il timeout, è quella in cui istanzio l'inputstream, non quella dove faccio readObject().
...

No, l'errore è dove fai l'operazione di read.
L'eccezione è chiarissima:

java.net.SocketTimeoutException: Read timed out


Quel time out di lettura c'entra proprio con il timeout che hai impostato sulla Socket con il meotod setSoTimeout, vedi javadoc:

public void setSoTimeout(int timeout)
throws SocketException

Enable/disable SO_TIMEOUT with the specified timeout, in milliseconds. With this option set to a non-zero timeout, a read() call on the InputStream associated with this Socket will block for only this amount of time. If the timeout expires, a java.net.SocketTimeoutException is raised, though the Socket is still valid. The option must be enabled prior to entering the blocking operation to have effect. The timeout must be > 0. A timeout of zero is interpreted as an infinite timeout.

Prova a leggere cos'è e cosa fa il metodo available di un InputStream, potrebbe tornarti utile per risolvere parzialmente il problema.

Il fatto qui è che quando il client tenta di leggere lo stream di input dal server, i dati non ci sono ne arrivano entro il timeout che gli hai impostato. E i motivi possono essere due (devi verificare quale è il tuo caso):
1) Il server deve ancora inviarli e comunque non arrivano entro il timeout di lettura impostato.
2) Il server li ha già inviati sullo stream, ma poi ha chiuso la connessione e a seguito di questa azione i dati inviati nello stream sono stati scartati: quando il client va a leggere non trova nulla, attende, scade il timeout di lettura e parte l'eccezione.

Il punto (2) può verificarsi, lo dice il javadoc di Socket.getInputStream():

public InputStream getInputStream()
throws IOException
...
Under abnormal conditions the underlying connection may be broken by the remote host or the network software (for example a connection reset in the case of TCP connections). When a broken connection is detected by the network software the following applies to the returned input stream :

* The network software may discard bytes that are buffered by the socket. Bytes that aren't discarded by the network software can be read using read.

* If there are no bytes buffered on the socket, or all buffered bytes have been consumed by read, then all subsequent calls to read will throw an IOException.

* If there are no bytes buffered on the socket, and the socket has not been closed using close, then available will return 0.

Closing the returned InputStream will close the associated socket.
...


In sinstesi, da quello che capisco, il tuo problema poterbbe essere la mancanza di un protocollo (per quanto minimale) tra client e server per gestire la comunicazione correttamente.