前言
最近要在安卓里面下载大文件,考虑到现场网络和使用的情况,如果简单http下载的话,很有可能下载崩溃或者中断失败。
原本想用wget做断点续传的,这样就简单很多,但是安卓不给运行wget
一、什么是分段下载?
简单来说就是把文件分拆开来,一块一块的下载,这样速度会有所损失,但胜在可控、健壮
在http里面,主要是请求头:Range
下面看一下Range字段常用表示的写法:
Range: bytes=0-1024 获取最前面1025个字节
Range: bytes=-500 获取最后500个字节
Range: bytes=1025- 获取从1025开始到文件末尾所有的字节
Range: 0-0 获取第一个字节
Range: -1 获取最后一个字节
例如,在一个请求头中有Range:byte=0-1024,那么表示的意思就是请求数据的前1025个字节。
如果这个分段请求的返回码是206,并且指示的分段范围是0-1024,文件的总大小是7877,那么在响应头中的数据应该表示为:
Content-Range: bytes 0-1024/7877
二、使用方法
直接上代码
package com.kl;
import com.kl.utils.IOUtils;
import com.kl.utils.Logx;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
class _DownLoadThread implements Runnable
{
private String _url = "";
private String _local = "";
private long _len = 0;
public _DownLoadThread(String url, String local, long len)
{
_url = url;
_local = local;
_len = len;
new Thread(this).start();
}
@Override
public void run()
{
long start, end, n = 0;
long size = 1024 * 1024 * 50;
start = 0;
_len--;
Logx.log("load: " + _url);
while (start < _len)
{
Logx.log("load process: " + start);
end = Math.min(start + size-1, _len);
if( down(start, end) )
{
n = 0;
start += size;
}
else n++;
if( n > 10 )
{
Logx.log("error, retry too many: " + _url);
break;
}
}
if( n==0 )
{
Logx.log("load complete");
NoticeUIChange.ins().newOk();
}
NoticeUIChange.ins().removeNew();
}
private boolean down(long startIndex, long endIndex)
{
HttpURLConnection conn = null;
InputStream is = null;
RandomAccessFile raf = null;
try
{
URL url = new URL(_url);
conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
//请求服务器下载部分的文件的指定位置
conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex);
conn.setConnectTimeout(5000);
is = conn.getInputStream();//返回资源
raf = new RandomAccessFile(_local, "rwd");
raf.seek(startIndex);//定位文件
int len = 0;
byte[] buffer = new byte[1024];
while ((len = is.read(buffer)) != -1) {
raf.write(buffer, 0, len);
}
is.close();
raf.close();
is = null;
raf = null;
return true;
}
catch (Exception e)
{
e.printStackTrace();
}
finally
{
if( is != null ) IOUtils.closeQuietly(is);
if(raf != null) IOUtils.closeQuietly(raf);
if( conn != null )
{
try{ conn.disconnect(); } catch (Exception e){}
}
}
return false;
}
}
里面的Logx类是日志类,可以都删掉或者替换自己的
IOUtils:
public class IOUtils
{
public static void closeQuietly(Closeable close)
{
if( close != null )
{
try
{
close.close();
}
catch (Exception er)
{
}
}
}
}
总结
本文实现的是单线程下载,考虑现场的情况,多线程反而会出问题。如果想要多现场下载,改装下下载类就行