Java TCP简单通信会产生意外结果

时间:2011-10-30 16:20:56

标签: java tcp hex

我试图在我的java程序中通过TCP发送一些简单的数据:

         String data = "70798090999a9b9c9d9e9fa0a1a2";
             ServerSocket srvr = new ServerSocket(1234);
             Socket skt = srvr.accept();
             OutputStream out = skt.getOutputStream();
             out.write(hexStringToByteArray(data));
             out.close();
             skt.close();
             srvr.close();

有功能:

public static byte[] hexStringToByteArray(String s) {
    int len = s.length();
    byte[] data = new byte[len / 2];
    for (int i = 0; i < len; i += 2) {
        data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4)
                             + Character.digit(s.charAt(i+1), 16));
    }
    return data;
}

并接收数据:

     Socket skt = new Socket("localhost", 1234);
     BufferedReader in = new BufferedReader(new
        InputStreamReader(skt.getInputStream()));

     while (!in.ready()) {}
     while (in.ready()) {
     System.out.println(in.read()); // Read one line and output it
     }

但是我没有收到正常增加的数字列表,而是随心所欲地增加和减少了一些东西:

121
196
234
244
246
245
250
249
251
252
8224
176
162

我在哪里做错了什么?

3 个答案:

答案 0 :(得分:3)

您正在发送字节数组并仅读取一个字节。

您必须使用如下所示的循环阅读:

byte[] arr = new byte[1024];
while (in.read(arr) >= 0) {
    // do something with the data.
}

这样可行,但仍然不是你真正需要的。真的,你用字符串操作,然后手动将字符串转换为字节数组,然后读取字节,以便再次手动将它们转换为字符串?

相反,您应该使用PrintWriter来编写字符串:

PrintWriter writer = new PrintWriter(new OutputStreamWriter(out));
writer.write(str);

然后在读取数据时使用BufferedReader。

BufferedReader reader = new BufferedReader(new InputStreamReader(in));
String line = reader.readLine();

答案 1 :(得分:3)

如果你想发送字节,我建议你读取字节,如果你想发送文字,我建议你阅读文字。你不应该尝试混合搭配,否则你可能会让自己感到困惑。

byte[] bytes = new BigInteger("70798090999a9b9c9d9e9fa0a1a2", 16).toByteArray();

ServerSocket ss = new ServerSocket(1234);
Socket c = new Socket("localhost", 1234);
Socket s = ss.accept();
final OutputStream out = s.getOutputStream();
out.write(bytes.length);
out.write(bytes);

final DataInputStream in = new DataInputStream(c.getInputStream());
int length = in.read();
byte[] bytes2 = new byte[length];
in.readFully(bytes2);

System.out.println(new BigInteger(1, bytes2).toString(16));
c.close();
s.close();
ss.close();

打印

70798090999a9b9c9d9e9fa0a1a2

答案 2 :(得分:1)

摆脱所有ready()测试。读取将阻塞,直到数据到达,因此调用ready()会浪费时间,浪费数据。