问题场景:如果两个socket通信,一个socket用从键盘获取一行数据,再通过打印流输出到另外一个socket,另外一个socket使用BufferedReader#readLine就可以接收到数据。这是正常的。
但是如果SocketChannel从键盘读取一行,把数据转换成字节数组写入到channl中,必须得加上"\r"(window系统是"\r\n"),另外一个socket才能接收到。
因为BufferedReader#readLine检测读取完一行必须是通过检测换行符才判断一行读取完成,所以SocketChannl写数据就要加上"\r\n",socket读取的时候就知道一行完成,所以readLine才会返回,否则一直阻塞。
本人怀疑是Socket通过输出流自动给数据加上了换行符,而SocketChannel通过通道发送 字节数组消息并没有加上换行符,所以需要手动加上换行符,readLine才可以检测到换行符,知道一行数据接收完成了,就返回。
经过一段时间排查,发现了正真的原因:
我用Socket通信时,把Socket输出流转换成了打印流PrintStream,然后调用PrintStream#println(str)把数据发给另外一个Socket。这个println()方法自动加了换行符,所以另一端Socket调用readLine()可以接受到。但是如果发送消息调用的是printStream#print(str),因为数据中没有换行符,所以要手动加上"\r",另外一个Socket才可通过readLine()接收到。