我正在创建一个服务器来接收来自客户端的文本和二进制数据。它适用于文本数据以及第一次接收二进制文件,但在此之后它不会继续读取数据并抛出异常。 这是我的服务器代码:
public class ConnectedProcessThread implements Runnable{
private final Socket socket;
public ConnectedProcessThread(Socket clientSocket){
socket = clientSocket;
}
public void run(){
DataInputStream dis = null;
try{
while(true) {
dis = new DataInputStream(socket.getInputStream());
String meta = dis.readUTF();
Log.i("Data received", meta);
if(meta.equalsIgnoreCase("Text")){
String message = dis.readUTF();
Log.i("Data received", message);
}else if(meta.equalsIgnoreCase("Binary")){
InputStream is = socket.getInputStream();
ByteArrayOutputStream stream = new ByteArrayOutputStream();
byte[] buf = new byte[4096];
int len;
while((len=is.read(buf))>-1){
stream.write(buf,0,len);
}
stream.flush();
//read object input
try {
ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(stream.toByteArray()));
byte[] buffer = (byte[])ois.readObject();
FileOutputStream fos = new FileOutputStream("/storage/emulated/0/DCIM/IMG-Saved.jpeg");
fos.write(buffer);
}catch (ClassNotFoundException e){
e.printStackTrace();
}
finally {
Log.i("Binary_Transfer","File created");
}
}
}
} catch (IOException e){
e.printStackTrace();
}finally {
Log.i("Client_Socket","Stream will close");
if(dis!=null){
try {
dis.close();
}catch (IOException e){
e.printStackTrace();
}
}
}
}
}
对于文本和二进制数据,在发送数据之前,客户端将发送文本元数据以通知服务器数据是文本还是二进制。但是在收到文件(图像)后,它会在行处抛出一个EOFException:String meta = dis.readUTF();我猜对了,因为在读取和写入二进制文件之后,线程继续循环,因此DataInputStream将再次读取,现在没有任何东西可以接收,因此readUTF()抛出EOFException。我发送二进制文件之后尝试从客户端发送元数据,让DataInputStream可以读取内容而不抛出异常,但是它不起作用,客户端确实发送了元数据,但服务器仍然抛出EOFException。谁知道问题是什么?非常感谢。
这是我从客户端发送的二进制方法:
public void sendBinaryData(byte[] binaryData){
if(dos!=null && socket!=null){
try {
ObjectOutputStream oos = new ObjectOutputStream(socket.getOutputStream());
oos.writeObject(binaryData);
Log.d("Binary_Transfer", "C: Sent.");
oos.close();
dos.writeUTF("Binary_End");
dos.flush();
}catch (Exception e){
Log.e("File_Exception",e.toString());
}
}
}
答案 0 :(得分:1)
因为当您开始读取二进制数据时,您输入的循环仅在流的末尾终止,即当对等方断开连接时:
while((len=is.read(buf))>-1){
stream.write(buf,0,len);
}
此时你就在流的最后。没有更多的数据,没有更多的数据,也没有更多的数据。
您需要完全删除此部分:
InputStream is = socket.getInputStream();
ByteArrayOutputStream stream = new ByteArrayOutputStream();
byte[] buf = new byte[4096];
int len;
while((len=is.read(buf))>-1){
stream.write(buf,0,len);
}
stream.flush();
无论如何,将内容读入ByteArrayOutputStreams
几乎没有任何意义,这也不例外。只需完全删除它,然后更改下一部分:
ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(stream.toByteArray()));
到
ObjectInputStream ois = new ObjectInputStream(socket.getInputStream());
并继续,因为你已经。但是你有另一个问题:
oos.close();
这会关闭套接字。所以下一部分:
dos.writeUTF("Binary_End");
dos.flush();
不可能奏效。只需清空ObjectOutputStream
而不是关闭它。
但我强烈建议您丢弃DataInput/OutputStreams
并在套接字的生命周期中使用单个ObjectInput/OutputStream
。