繁体   English   中英

Java TCP简单通信带来了意想不到的结果

[英]Java TCP simple communication gives unexpected results

我正在尝试在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

我在哪里做错了什么?

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

您必须阅读using循环,如下所示:

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

这将起作用,但仍然不是您真正需要的。 确实,您使用字符串进行操作,但是随后将字符串手动转换为字节数组,然后读取字节以再次将它们手动转换为String?

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

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

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

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

如果要发送字节,我建议您阅读字节,如果要发送文本,我建议您阅读文本。 您不应该尝试混合搭配,否则可能会感到困惑。

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

摆脱所有ready()测试。 读取将阻塞,直到数据到达为止,因此调用ready()确实浪费时间,并且有丢失数据的风险。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM