在socket通信中,经常会有传送大量字节数组的时候,在一般情况下,直接用read(byte[] b)读取数组不会出现问题,但是当文件传输速度很快,或者量特别大的情况下。就可能导致传送的字节不完整。原因如下:
由于我们用的是基于TCP/IP的socket传输。发送端的数据首先进入缓存区被分割成TCP/IP报文,然后排队进入管道中,通过管道在进入接收端的缓存区进行排序再进入上层处理。由于数据在管道中的速度不同导致接收端的数据顺序根据tcp/ip协议需要重新排列。这时问题就来了,由于数据并不是按顺序一个个通过管道过来的,可能当一部分数据被传送过来,而还有数据并没有传输完毕。这时候使用read(byte[] b)去读写文件。这里请开jdk源码
public int read(byte b[], int off, int len) throws IOException {
if (b == null) {
throw new NullPointerException();
} else if (off < 0 || len < 0 || len > b.length - off) {
throw new IndexOutOfBoundsException();
} else if (len == 0) {
return 0;
}
int c = read();
if (c == -1) {
return -1;
}
b[off] = (byte)c;
int i = 1;
try {
for (; i < len ; i++) {
c = read();
if (c == -1) {
break;
}
b[off + i] = (byte)c;
}
} catch (IOException ee) {
}
return i;
可以看出这里的read(byte[] b)是经过for循环然后read来达到读取整个数组的目的的。但是由于数据的分批到达的性质,可能在一段时间内数据没有及时到达,这是read()方法就会返回-1,而read(byte[] b)也就直接跳出了。所得到的byte数组也就没有完全写入,结果就丢失了一部分文件,继而后续也就都出现了错误。而解决方法可能有很多,这里先介绍一种简单的,可能后续再补充其他方法。使用dataInputStream中readFully(byte[] b)方法读取数组。看其源码:
public final void readFully(byte b[], int off, int len) throws IOException {
if (len < 0)
throw new IndexOutOfBoundsException();
int n = 0;
while (n < len) {
int count = in.read(b, off + n, len - n);
if (count < 0)
throw new EOFException();
n += count;
}
}
由源码可知,其中使用while循环判断byte数组是否已经读满所有数据,如果没有读满则继续读取补充直到读满为止,从而改善输入流出现空档,造成read方法直接跳出的问题。即通过缓冲来保证数量的完整,也算是常用的一种方法。所以以后若要读取特定长度的数据,使用readFully读取更加安全。