`BufferedInputStream.available()`是否可以安全地获取已经收到的字节?

时间:2013-01-09 09:46:09

标签: java sockets tcp

我希望从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 /什么的线程? (需要同步)

我的应用程序需要是实时的(充其量,但是毫秒级的延迟基本上可以作为休闲程序使用),所以我非常担心。该计划大部分都是不完整的,所以我还不能尝试。

1 个答案:

答案 0 :(得分:0)

  

可以“无阻塞地读取的字节”可以解释为“套接字已经收到的字节”吗?

这是唯一的方式。

  

请问istream.available()实际返回的数字是否少于已收到的数据(即某些数据尚未发送到缓冲区)?

即使是未决数据,它也可以从零向上返回任何内容。在SSLSockets上,它总是返回零。

但是您不需要调用它,并且您不需要所有精心设计的代码。只需在阻塞模式下读取流,并使用多个线程。