天天看点

关于 InputStream 类的 available() 方法

API 对 available() 方法说明:

/**
* 返回此输入流下一个方法调用可以不受阻塞地从此输入流读取(或跳过)的估计字节数。
* 下一个调用可能是同个线程,也可能是另一个线程。
* 一次读取或跳过此估计数个字节不会受阻塞,但读取或跳过的字节数可能小于该数。
*/
public int available() throws IOException
           

要一次读取多个字节时,经常用到 InputStream.available() 方法,这个方法可以在读写操作前先得知数据流里有多少个字节可以读取。需要注意的是,如果这个方法用在从本地文件读取数据时,一般不会遇到问题,但如果是用于网络操作,就经常会遇到一些麻烦。比如,Socket 通讯时,对方明明发来了1000个字节,但是自己的程序调用 available() 方法却只得到900,或者100,甚至是0,感觉有点莫名其妙,怎么也找不到原因。其实,这是因为网络通讯往往是间断性的,一串字节往往分几批进行发送。本地程序调用 available() 方法有时得到0,这可能是对方还没有响应,也可能是对方已经响应了,但是数据还没有送达本地。对方发送了1000个字节给你,也许分成3批到达,这你就要调用3次 available() 方法才能将数据总数全部得到。

如果这样写代码:

  int count = in.available();
  byte[] b = new byte[count];
  in.read(b);
// 在进行网络操作时往往出错,因为你调用available()方法时,对发发送的数据可能还没有到达,你得到的count是0。
           

需要改成这样:

  int count = 0;
  while (count == 0) {
   count = in.available();
  }
  byte[] b = new byte[count];
  in.read(b);
           
能否使用取决于实现了InputStream这个抽象类的具体子类中有没有实现available这个方法。
如果实现了那么就可以取得大小,如果没有实现那么就获取不到。
例如FileInputStream就实现了available方法,那么就可以用new byte[in.available()];这种方式。
但是,网络编程的时候Socket中取到的InputStream,就没有实现这个方法,那么就不可以使用这种方式创建数组。
           

FileInputStream 中已经重写了这个方法,重写之后的方法是通过文件的描述来获取文件的大小的。所以就 FileInputStream  可以用 new byte[in.available()]; 这种方式:

// 待处理的图片
String imgFile = imgPath;
InputStream in = null;
byte[] data = null;
// 读取图片字节数组
in = new FileInputStream(imgFile);
data = new byte[in.available()];
in.read(data);
           

参考来源于:

http://www.cnblogs.com/MyFavorite/archive/2010/10/19/1855758.html

https://www.cnblogs.com/lm970585581/p/7257146.html

继续阅读