返回此输入流下一个方法调用可以不受阻塞地从此输入流读取(或跳过)的估计字节数。下一个调用可能是同一个线程,也可能是另一个线程。一次读取或跳过此估计数个字节不会受阻塞,但读取或跳过的字节数可能小于该数。
注意,有些 InputStream 的实现将返回流中的字节总数,但也有很多实现不会这样做。试图使用此方法的返回值分配缓冲区,以保存此流所有数据的做法是不正确的。
如果已经调用 close() 方法关闭了此输入流,那么此方法的子类实现可以选择抛出 IOException。
类 InputStream 的 available 方法总是返回 0。
此方法应该由子类重写。
返回: 可以不受阻塞地从此输入流读取(或跳过)的估计字节数;如果到达输入流末尾,则返回 0。 抛出:
inputStream 源代码
/*** Returns the number of bytes that are available before this stream will
* block. This implementation always returns 0. Subclasses should override
* and indicate the correct number of bytes available.
*
*@returnthe number of bytes available before blocking.
*@throwsIOException
* if an error occurs in this stream.
*@sinceAndroid 1.0*/
public int available() throwsIOException {return 0;
}
这里返回的是 0 值。
所以说要从网络中下载文件时,我们知道网络是不稳定的,也就是说网络下载时,read()方法是阻塞的,说明这时我们用
inputStream.available()获取不到文件的总大小。
但是从本地拷贝文件时,我们用的是FileInputStream.available(),难道它是将先将硬盘中的数据先全部读入流中?
然后才根据此方法得到文件的总大小?
好吧,我们来看看FileInputStream源代码吧
/*** Returns the number of bytes that are available before this stream will
* block. This method always returns the size of the file minus the current
* position.
*
*@returnthe number of bytes available before blocking.
*@throwsIOException
* if an error occurs in this stream.
*@sinceAndroid 1.0*/@Overridepublic int available() throwsIOException {
openCheck();//BEGIN android-added//Android always uses the ioctl() method of determining bytes//available. See the long discussion in//org_apache_harmony_luni_platform_OSFileSystem.cpp about its//use.
returnfileSystem.ioctlAvailable(fd.descriptor);//END android-added//BEGIN android-deleted//synchronized (repositioningLock) {// //stdin requires special handling//if (fd == FileDescriptor.in) {//return (int) fileSystem.ttyAvailable();//}//
//long currentPosition = fileSystem.seek(fd.descriptor, 0L,//IFileSystem.SEEK_CUR);//long endOfFilePosition = fileSystem.seek(fd.descriptor, 0L,//IFileSystem.SEEK_END);//fileSystem.seek(fd.descriptor, currentPosition,//IFileSystem.SEEK_SET);//return (int) (endOfFilePosition - currentPosition);//}//END android-deleted
}
这里重写了inputStream中的available()方法
关键是:fileSystem.ioctlAvailable(fd.descriptor);
调用了FileSystem这是java没有公开的一个类,JavaDoc API没有。
其中
fileSystem 是一个IFileSystem对象,IFileSySTEM是java没有公开的一个类,JavaDoc API中没有;
fd是一个FileDescriptor对象,即文件描述符
说明这句代码应该是通过文件描述符获取文件的总大小,而并不是事先将磁盘上的文件数据全部读入流中,再获取文件总大小
搞清楚了这些,但是我们的主要问题没有解决,怎么获得网络文件的总大小?
我想原理应该都差不多,应该也是通过一个类似文件描述符的东西来获取。
网络下载获取文件总大小的代码:
HttpURLConnection httpconn =(HttpURLConnection)url.openConnection();
httpconn.getContentLength();
我们再来看看httpconn.getContentLength();
/*** Gets the content length in bytes specified by the response header field
* {@codecontent-length} or {@code-1} if this field is not set.
*
*@returnthe value of the response header field {@codecontent-length}.
*@sinceAndroid 1.0*/
public intgetContentLength() {return getHeaderFieldInt("Content-Length", -1);
}
关键:getHeaderFieldInt("Content-Length", -1);
意思是从http预解析头中获取“Content-length”字段的值
其实也是类似从文件描述符中获取文件的总大小