转自:http://hold-on.iteye.com/blog/1017449

如果用inputStream对象的available()方法获取流中可读取的数据大小,通常我们调用这个函数是在下载文件或者对文件进行其他处理时获取文件的总大小。

以前在我们初学File和inputStream和outputStream时,有需要将文件从一个文件夹复制到另一个文件夹中,这时候我们用的就是inputStream.available()来获取文件的总大小,而且屡试不爽。

但是当我们要从网络URL中下载一个文件时,我们发现得到的数值并不是需要下载的文件的总大小。

好吧。我们看看JDK文档中怎么解释。

available

public int available()
throws IOException

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

注意,有些 InputStream 的实现将返回流中的字节总数,但也有很多实现不会这样做。试图使用此方法的返回值分配缓冲区,以保存此流所有数据的做法是不正确的。

如果已经调用 close() 方法关闭了此输入流,那么此方法的子类实现可以选择抛出 IOException

类 InputStream 的 available 方法总是返回 0

此方法应该由子类重写。

返回:可以不受阻塞地从此输入流读取(或跳过)的估计字节数;如果到达输入流末尾,则返回 0抛出:IOException - 如果发生 I/O 错误。

inputStream 源代码

  1. /**
  2. * Returns the number of bytes that are available before this stream will
  3. * block. This implementation always returns 0. Subclasses should override
  4. * and indicate the correct number of bytes available.
  5. *
  6. * @return the number of bytes available before blocking.
  7. * @throws IOException
  8. *             if an error occurs in this stream.
  9. * @since Android 1.0
  10. */
  11. <span style="color: #ff0000;"> public int available() throws IOException {
  12. return 0;
  13. }</span>

这里返回的是 0 值。

所以说要从网络中下载文件时,我们知道网络是不稳定的,也就是说网络下载时,read()方法是阻塞的,说明这时我们用

inputStream.available()获取不到文件的总大小。

但是从本地拷贝文件时,我们用的是FileInputStream.available(),难道它是将先将硬盘中的数据先全部读入流中?

然后才根据此方法得到文件的总大小?

好吧,我们来看看FileInputStream源代码吧

  1. /**
  2. * Returns the number of bytes that are available before this stream will
  3. * block. This method always returns the size of the file minus the current
  4. * position.
  5. *
  6. * @return the number of bytes available before blocking.
  7. * @throws IOException
  8. *             if an error occurs in this stream.
  9. * @since Android 1.0
  10. */
  11. @Override
  12. public int available() throws IOException {
  13. openCheck();
  14. // BEGIN android-added
  15. // Android always uses the ioctl() method of determining bytes
  16. // available. See the long discussion in
  17. // org_apache_harmony_luni_platform_OSFileSystem.cpp about its
  18. // use.
  19. <span style="color: #ff0000;">return fileSystem.ioctlAvailable(fd.descriptor);</span>
  20. // END android-added
  21. // BEGIN android-deleted
  22. // synchronized (repositioningLock) {
  23. //     // stdin requires special handling
  24. //     if (fd == FileDescriptor.in) {
  25. //         return (int) fileSystem.ttyAvailable();
  26. //     }
  27. //
  28. //     long currentPosition = fileSystem.seek(fd.descriptor, 0L,
  29. //             IFileSystem.SEEK_CUR);
  30. //     long endOfFilePosition = fileSystem.seek(fd.descriptor, 0L,
  31. //             IFileSystem.SEEK_END);
  32. //     fileSystem.seek(fd.descriptor, currentPosition,
  33. //             IFileSystem.SEEK_SET);
  34. //     return (int) (endOfFilePosition - currentPosition);
  35. // }
  36. // END android-deleted
  37. }

这里重写了inputStream中的available()方法

关键是:fileSystem.ioctlAvailable(fd.descriptor);

调用了FileSystem这是java没有公开的一个类,JavaDoc API没有。
其中

fileSystem 是一个IFileSystem对象,IFileSySTEM是java没有公开的一个类,JavaDoc API中没有;

fd是一个FileDescriptor对象,即文件描述符

说明这句代码应该是通过文件描述符获取文件的总大小,而并不是事先将磁盘上的文件数据全部读入流中,再获取文件总大小

搞清楚了这些,但是我们的主要问题没有解决,怎么获得网络文件的总大小?

我想原理应该都差不多,应该也是通过一个类似文件描述符的东西来获取。

网络下载获取文件总大小的代码:

  1. <span style="color: #ff0000;">HttpURLConnection httpconn = (HttpURLConnection)url.openConnection();
  2. httpconn.getContentLength();</span>

我们再来看看httpconn.getContentLength();

  1. /**
  2. * Gets the content length in bytes specified by the response header field
  3. * {@code content-length} or {@code -1} if this field is not set.
  4. *
  5. * @return the value of the response header field {@code content-length}.
  6. * @since Android 1.0
  7. */
  8. public int getContentLength() {
  9. <span style="color: #ff0000;">return getHeaderFieldInt("Content-Length", -1);</span> //$NON-NLS-1$
  10. }

关键:getHeaderFieldInt("Content-Length", -1);

意思是从http预解析头中获取“Content-length”字段的值

其实也是类似从文件描述符中获取文件的总大小

最新文章

  1. C#7.0中有哪些新特性?
  2. [Python] python vs cplusplus
  3. java mkdir()和mkdirs()的区别
  4. WebStorm中将Project分享到GitHub时报“Error Running Git”错误的解决办法
  5. Redis系列文章导读
  6. 45种Javascript技巧大全(转)
  7. 访谈将源代码的函数 strcpy/memcpy/atoi/kmp/quicksort
  8. Hashtable和HashMap类
  9. Unity3D Asset Server搭建 .
  10. git下载/上传文件提示:git did not exit cleanly
  11. c++ 动态生成string类型的数组
  12. 第一节,windows和ubuntu下深度学习theano环境搭建
  13. 在同一个服务器(同一个IP)为不同域名绑定的免费SSL证书
  14. android -------- Data Binding的使用 ( 四 )ListView
  15. Utils--字符串的帮助类
  16. 短小而精悍的JsvaScript函数
  17. 助你简化开发的 jQuery 插件
  18. srTCM和trTCM介绍
  19. Android之ConnectivityManager
  20. Swift学习——A Swift Tour 协议和扩展

热门文章

  1. Cent os FTP配置
  2. conso.log占位符
  3. Word截图PNG,并压缩图片大小
  4. ofbiz之entity实体写法
  5. MVC - 13.验证
  6. JAVA用POI读取和创建2003和2007版本Excel
  7. Table上下滚动
  8. [BZOJ5305][Haoi2018]苹果树 组合数
  9. Child Action
  10. 封装CSS动画