您的位置:首页 > 大数据 > 人工智能

关于inputStream.available()方法获取下载文件的总大小

2014-08-05 20:47 495 查看


关于inputStream.available()方法获取下载文件的总大小

博客分类:

java

AndroidJDKApache

Java代码


如果用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 源代码

Java代码


/**

* 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.

*

* @return the number of bytes available before blocking.

* @throws IOException

* if an error occurs in this stream.

* @since Android 1.0

*/

<span style="color: #ff0000;"> public int available() throws IOException {

return 0;

}</span>

这里返回的是 0 值。

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

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

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

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

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

Java代码


/**

* 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.

*

* @return the number of bytes available before blocking.

* @throws IOException

* if an error occurs in this stream.

* @since Android 1.0

*/

@Override

public int available() throws IOException {

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.

<span style="color: #ff0000;">return fileSystem.ioctlAvailable(fd.descriptor);</span>

// 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对象,即文件描述符

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

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

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

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

Java代码


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

httpconn.getContentLength();</span>

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

Java代码


/**

* Gets the content length in bytes specified by the response header field

* {@code content-length} or {@code -1} if this field is not set.

*

* @return the value of the response header field {@code content-length}.

* @since Android 1.0

*/

public int getContentLength() {

<span style="color: #ff0000;">return getHeaderFieldInt("Content-Length", -1);</span> //$NON-NLS-1$

}

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

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

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

在网络编程中,我的习惯是先用InputStream.available()方法来检查是否有可用的数据可以读,如果没有,我就先等着,等有了以后,我再读取。

在等的过程中,我还可以进行超时检测,代码如下所示。

但是最近的项目中遇到一个问题,就是available()方法一直返回的是0,我用BufferedReader.ready()就可以读回来,大家知道是怎么回事儿吗?或者大家都是如何写这种接收的程序呢?

代码如下:

Java代码


int len = is.available();

long t1 = System.currentTimeMillis();

while (len == 0) {

long t2 = System.currentTimeMillis();

// 检查接收是否超时,如果超时就抛出异常

if (this.soTimeout < (t2-t1))

throw new TimeoutException("接收数据超时:" + (t2-t1) + "ms");

Thread.currentThread().sleep(10);

len = is.available();

}

byte[] bs = new byte[len];

this.is.read(bs);

声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。

推荐链接

返回顶楼
gccr
等级: 初级会员



性别:


文章: 4
积分: 40
来自: 北京



发表时间:2011-08-31

在网上看到有说必须要先read(),然后再调用available()方法来检测可以读的数据大小,但是如果真这样操作,对方一直不给发数据,我这边就会一直等,超时不好控制呀。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: