我希望从BufferedInputStream istream
的流中收集来自Socket
的一些数据。数据首先包含一个字节n
(8位无符号整数),然后是n
个字节,其中包含实际消息(二进制格式,不是字符串)。所以我编写了以下代码(未经过测试):
public byte[] getNextMessageBytes() throws IOException {
// Make sure the "length" byte is ready to be read
if (this.istream.available() < 1) {
return null;
}
// Peek 1 byte
this.istream.mark(1);
int length = this.istream.read();
// -1 == EOF
if (length < 0) {
throw new EOFException();
}
// reset the marked byte
this.istream.reset();
// If all bytes available...
// NOTE the 'length' byte is still in the stream
if (this.istream.available() > length) {
// Peek the bytes...
this.istream.mark(length + 1);
// Skips the 'length' byte
this.istream.skip(1);
byte[] data = new byte[length];
int recv = this.istream.read(data, 0, length);
if (recv == length) {
this.istream.mark(0);
return data;
}
// -1 == EOF
if (recv < 0) {
throw new EOFException();
}
this.istream.reset();
}
return null;
}
此代码基本上首先读取长度为n
的字节。然后,如果istream.available()
表示可用的全长,它将尝试读取一定数量的字节,如果成功,则返回字节数组。
docs of BufferedInputStream.available()
:
通过下次调用此输入的方法,返回估计可以从此输入流读取(或跳过)而不阻塞的字节数流。下一次调用可能是同一个线程或另一个线程。单个读取或跳过这么多字节不会阻塞,但可以读取或跳过更少的字节。
可以“无阻塞地读取的字节”可以解释为“套接字已经收到的字节”吗? istream.available()
实际上会返回比已经收到的数据更少的数字(即一些尚未发送到缓冲区的数据)吗?
或者它实际上是一个更好的主意,开始一个继续阻塞读取和推送消息到队列/列表/向量/ arraylist /什么的线程? (需要同步)
我的应用程序需要是实时的(充其量,但是毫秒级的延迟基本上可以作为休闲程序使用),所以我非常担心。该计划大部分都是不完整的,所以我还不能尝试。
答案 0 :(得分:0)
可以“无阻塞地读取的字节”可以解释为“套接字已经收到的字节”吗?
这是唯一的方式。
请问istream.available()实际返回的数字是否少于已收到的数据(即某些数据尚未发送到缓冲区)?
即使是未决数据,它也可以从零向上返回任何内容。在SSLSockets上,它总是返回零。
但是您不需要调用它,并且您不需要所有精心设计的代码。只需在阻塞模式下读取流,并使用多个线程。