为什么DataInputStream在读取二进制文件后不再继续接收数据

时间:2014-09-01 03:07:41

标签: java bytearray binaryfiles objectinputstream datainputstream

我正在创建一个服务器来接收来自客户端的文本和二进制数据。它适用于文本数据以及第一次接收二进制文件,但在此之后它不会继续读取数据并抛出异常。 这是我的服务器代码:

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());
        }
    }
}

1 个答案:

答案 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