1

我正在尝试使用 GZIPOutputStream 在客户端中对字符串进行编码,然后使用 GZIPOutputStream 在服务器中解码该字符串。

客户端代码(初始套接字连接建立后)为:

// ... Establishing connection, getting a socket object.
// ... Now proceeding to send data using that socket:

DataOutputStream out = new DataOutputStream(socket.getOutputStream());
String message = "Hello World!";

ByteArrayOutputStream out = new ByteArrayOutputStream();
GZIPOutputStream gzip = new GZIPOutputStream(out);
gzip.write(message);
gzip.close();
String encMessage = out.toString();

out.writeInt(encMessage.getBytes().length);
out.write(encMessage.getBytes());
out.flush();

以及服务器端代码(同样,在建立连接之后):

DataInputStream input = new DataInputStream(socket.getInputStream());

int length = input.readInt();
byte[] buffer = new byte[length];
input.readFully(buffer);

GZIPInputStream gz = new GZIPInputStream(new ByteArrayInputStream(buffer));
BufferedReader r = new BufferedReader(new InputStreamReader(gz));
String s = "";
String line;
while ((line = r.readLine()) != null) 
{
    s += line;
}

我检查并正确传递了缓冲区长度(即编码消息的大小),因此传输了正确的字节数。但是,我得到了这个:

java.util.zip.ZipException: invalid code lengths set
at java.util.zip.InflaterInputStream.read(InflaterInputStream.java:164)
at java.util.zip.GZIPInputStream.read(GZIPInputStream.java:117)
at java.util.zip.InflaterInputStream.read(InflaterInputStream.java:122)
at parsing.ReceiveResponsesTest$TestReceiver.run(ReceiveResponsesTest.java:147)
at java.lang.Thread.run(Thread.java:745)

有任何想法吗?

提前感谢您的任何帮助!

4

2 回答 2

4

您正在调用- 这是不正确toString()ByteArrayOutputStream,它会引发各种可能在此处困扰您的字符编码问题。您需要toByteArray改为:

byte[] encMessage = out.toByteArray();

out.writeInt(encMessage.length);
out.write(encMessage);

细节:

如果您使用toString(),Java 将以您的平台默认字符编码对您的字节进行编码。那可能是一些 Windows 代码页、UTF-8 或诸如此类的东西。然而,并非所有字符都可以正确编码,有些字符将被替换字符替换 - 也许是问号。不知道细节,很难说。

但无论如何,将字节数组编码为String,然后在写出来的时候再次解码为字节数组,很有可能会改变字节数组中的数据。并且不需要这样做,您可以直接获取字节数组,如上面的代码所示。

于 2017-02-06T05:04:43.920 回答
1

你到底为什么沉迷于所有这些复杂的事情中?您可以将其简化为:

GZIPOutputStream gzip = new GZIPOutputStream(socket.getOutputStream());
DataOutputStream out = new DataOutputStream(gzip);
String message = "Hello World!";    
out.writeUTF(message);
out.close();

// ...    

GZIPInputStream gz = new GZIPInputStream(new ByteArrayInputStream(socket.getInputStream()));
DataInputStream input = new DataInputStream(gz);
String line = input.readUTF();

我进一步注意到您的代码实际上并没有编译。我会进一步指出,除非消息大几个数量级,否则 GZipping 没有任何好处。

于 2017-02-06T08:46:29.047 回答