我正在尝试使用NIO实现一个简单的客户端 - 服务器应用程序。
作为练习,沟通应该是基于文本和面向行的。但是当服务器读取客户端发送的字节时,它什么都没有,或者更确切地说,缓冲区中充满了一堆零。
我正在使用选择器,当通道可读时,这是触发的代码。
private void handleRead() throws IOException {
System.out.println("Handler Read");
while (lineIndex < 0) {
buffer.clear();
switch (channel.read(buffer)) {
case -1:
// Close the connection.
return;
case 0:
System.out.println("Nothing to read.");
return;
default:
System.out.println("Converting to String...");
buffer.flip();
bufferToString();
break;
}
}
// Do something with the line read.
}
在此代码段中,lineIndex
是int
,其中包含读取时发生第一个\n
的索引。它使用-1
进行初始化,这意味着不存在\n
。
变量buffer
引用ByteBuffer
,channel
代表SocketChannel
。
为了简单起见,没有Charset
s等等,这就是bufferToString
的编码方式:
private void bufferToString() {
char c;
System.out.println("-- Buffer to String --");
for (int i = builder.length(); buffer.remaining() > 1; ++i) {
c = buffer.getChar();
builder.append(c);
System.out.println("Appending: " + c + "(" + (int) c + ")");
if (c == '\n' && lineIndex < 0) {
System.out.println("Found a new-line character!");
lineIndex = i;
}
}
}
变量builder
包含对StringBuilder
的引用。
我希望getChar
进行合理的转换,但我输出的所有内容都是一堆(相当于缓冲区容量的一半)
追加:(0)
由
终止无需阅读。
可能是什么原因的线索?我在客户端有类似的代码,也无法从服务器上正确读取任何内容。
如果有任何帮助,下面是编写代码的示例:
private void handleWrite() throws IOException {
buffer.clear();
String msg = "Some message\n";
for (int i = 0; i < msg.length(); ++i) {
buffer.putChar(msg.charAt(i));
}
channel.write(buffer);
}
我还确认channel.write
的结果大于零,确保字节确实已写入并发送。
答案 0 :(得分:2)
事实证明,这是一个缓冲区索引问题。在服务器中,在写入套接字之前缺少flip()
。在客户端代码中,在阅读之后和写入之前,还有一些flip()
也丢失了。现在一切都按预期工作了。
当前的编写代码(服务器端):
private void handleWrite() throws IOException {
String s = extractLine();
for (int i = 0, len = s.length(); i < len;) {
buffer.clear();
while (buffer.remaining() > 1 && i < len) {
buffer.putChar(s.charAt(i));
++i;
}
buffer.flip();
channel.write(buffer);
}
// some other operations...
}