0

你好,我正在开发一个使用Netty的服务器。目标是能够来回发送对象。所以我使用ObjectDecoderInputStream和ObjectEncoderOutputStream配置了服务器。

所以我有一个客户端也使用 Netty 与服务器通信。客户端工作得很好。

每当我使用 Java 的套接字从服务器接收一个可序列化的对象时。发生了一些奇怪的事情,我在下面遇到了一个异常:

java.io.StreamCorruptedException: Unsupported version: 0
at org.jboss.netty.handler.codec.serialization.CompactObjectInputStream.readStreamHeader(CompactObjectInputStream.java:38)
at java.io.ObjectInputStream.<init>(Unknown Source)
at org.jboss.netty.handler.codec.serialization.CompactObjectInputStream.<init>(CompactObjectInputStream.java:30)
at org.jboss.netty.handler.codec.serialization.ObjectDecoderInputStream.readObject(ObjectDecoderInputStream.java:115)
at com.bvd.main.Client.readResponse(Client.java:139)
at com.bvd.main.Client.getFile(Client.java:80)
at com.bvd.main.Client.main(Client.java:163)

从客户端发送对象:

    public static void writeRequest(Message requestMsg,
        OutputStream outputStream) {
    byte[] bytes = null;
    ByteArrayOutputStream byteOutputStream = new ByteArrayOutputStream();
    try {
        ObjectEncoderOutputStream objectOutputStream = new ObjectEncoderOutputStream(
                byteOutputStream, 8192);
        objectOutputStream.writeObject(requestMsg);
        objectOutputStream.flush();
        bytes = byteOutputStream.toByteArray();
        byteOutputStream.close();
        objectOutputStream.close();
        outputStream.write(bytes);
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

以及客户端收到对象的代码:

    public static Object readResponse(InputStream inputStream) {
    int count = 0;
    Object object = null;
    try {
        byte[] lenBuffer = new byte[4];
        while ((count = inputStream.read(lenBuffer)) < 4) {
        }
        System.out.println(ByteBuffer.wrap(lenBuffer).getInt());
        int infolen = ByteBuffer.wrap(lenBuffer).getInt();
        byte[] objBuffer = new byte[infolen];
        while ((count = inputStream.read(objBuffer)) < infolen) {
        }
        byte[] totalBuf = new byte[4 + infolen];
        System.arraycopy(lenBuffer, 0, totalBuf, 0, lenBuffer.length);
        System.arraycopy(objBuffer, 0, totalBuf, lenBuffer.length,
                objBuffer.length);

        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(
                totalBuf);
        System.out.println(byteArrayInputStream.available());
        ObjectDecoderInputStream objectDecoderInputStream = new ObjectDecoderInputStream(
                new BufferedInputStream(byteArrayInputStream));
        object = objectDecoderInputStream.readObject();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (ClassNotFoundException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    return object;
}

更让我困惑的是,当我在2858(byte)的服务器上设置发送的对象内容的长度(该对象携带部分已转换为byte[]的文件)时,上面的客户端运行良好;当我将长度更改为 2859 或更大时,上面会发生StreamCorruptedException

ps:发送对象工作正常,服务器接收成功并正确解码,所有问题都是关于从服务器接收对象的问题。

对不起我的英语不好,任何帮助表示赞赏。谢谢。

4

1 回答 1

0

如果您最终进行了多个读取调用,那么在您的两个读取循环中,您将获得虚假数据:

    while ((count = inputStream.read(lenBuffer)) < 4) {
    }


    while ((count = inputStream.read(objBuffer)) < infolen) {
    }

每次调用 read 时,它都会写入byte[]. 因此,每次循环时,都会覆盖缓冲区中的部分数据,而不是读取整个缓冲区的数据。您需要跟踪当前偏移量(基于您目前已读取的数据量)并将其传递给线程read()调用。另外,您应该累积计数,而不是分配给它。

于 2012-07-23T02:43:37.077 回答