1.介绍
慢慢造轮子,在造轮子的过程中学习android相关的原理和设计模式。先开始一步步实现多线程文件断点下载器。
这次的多线程文件断点下载器,要实现以下几点:
1. 断点续传,不只是单纯的点个暂停开始,而是在退出重进之后仍然有任务的进度,这个就需要用到数据持久化了。
2. 多任务并行下载,一定数量的任务并行下载,超过额定值的任务暂停等待。
3. 单任务多线程下载,这个需要服务端的支持
2.从零开始
这个从0开始不是说从新建工程开始,而是从简单的下载一个文件开始。
这次网络请求用的库是okhttp,当然用URLConnection也可以。
先来看一下下载文件的代码,很简单,新建一个线程进行下载文件的耗时操作。
@Override
public void run()
{
String url = ""; //文件的url地址
OkHttpClient client = new OkHttpClient(); //okhttp client
RandomAcessFile file;
BufferedInputStream bis = null; //读取okhttp响应的流
String fileSavePath = ""; //文件路径
try {
//构建okhttp的请求
Request request = new Request.Builder()
.url(url)
.build();
Response response = client.newCall(request).execute();
//获得响应内容
ResponseBody responseBody = response.body();
//做一个判断,如果返回为空,则不进行下面的下载
if (responseBody == null) {
System.out.println("resource not found");
return;
}
//不为空,获取流,并且创建文件流
bis = new BufferedInputStream(response.body().byteStream());
file = new RandomAcessFile(fileSavePath, "rwd");
//创建一个缓冲区,用来读写文件,设置一个合适的大小,防止占用过多内存
byte[] buff = new byte[5 * 1024];
int len;
while ((len = bis.read(buff)) > 0) {
file.write(buff, 0, len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
//关闭流
try {
if (fos != null)
fos.close();
if (bis != null)
bis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
这就完成了一个简单任务下载的过程,从指定的url处下载文件,并保存到SD卡。接下来开始尝试加入断点下载功能。
3.断点下载
先规划一下断点下载最基本的功能
- 可以控制任务切换下载状态,并从正确的地方继续下载(如果服务端不支持断点,则重新下载)
- 可以保存下载进度
- 可以判断服务端是否支持断点下载,并采用不同的策略
第一点,可以利用一组变量表示当前一个任务的下载状态,通过改变这个变量来间接控制任务,通过读取变量获得任务当前的状态。
public class LoadState
{
public static final int PREPARE=1;
public static final int START=2;
public static final int DOWNLOADING=3;
public static final int PAUSE=4;
public static final int NET_ERROR=5;
public static final int COMPLETED=6;
public static final int CANCEL=7;
}
因为没有同时拥有两种状态的情况,就用简单的值表示。
修改一下上面的代码,让任务过程和状态关联起来,并加入数据储存。这里使用的是SQLite,为了方便,用了GreenDao,生成代码入下
private static void addDownloadEntity(Schema schema)
{
Entity entity = schema.addEntity("DownloadEntity");
entity.addStringProperty("url").primaryKey();
entity.addLongProperty("taskSize");
entity.addLongProperty("completedSize");
entity.addStringProperty("saveDirPath");
entity.addStringProperty("fileName");
}
需要保存的主要有
protected long taskSize; //文件大小
protected long completedSize; //已完成大小
protected String url; //地址
protected String fileName; //文件名
protected String saveDirPath; //保存地址
每写入一段数据,就储存一下进度到数据库,这样即使网络突然中断数据也不会丢失。
断点续传的主要实现就是使用Http中的Range头来进行断点请求,令服务器返回Range头指定的范围。这里也可以判断服务器是否支持断点下载,支持断点下载的服务器响应里会有Content-Range头,如果不含Content-Range头,则就不需要保存进度,暂停后恢复需要重新下载。判断的主要方法如下。
String Content_Range = response.header("Content-Range");
if (Content_Range == null) {
//if not support , get content length and use single thread process download
taskSize = responseBody.contentLength();
} else
//else get length from head content range or
//directly get by content length(request use Range: byte:0-)
taskSize = Long.parseLong(Content_Range.substring(Content_Range.lastIndexOf("/") + 1));
保存进度的方法就是记录进度,在每读取一段数据之后将记录写入数据库,将开始代码的写入部分改成这样即可。
int len;
while ((len = bis.read(buffer)) > 0 && state == LoadState.DOWNLOADING) {
file.write(buffer, 0, length);
completedSize += length;
//更新进度并保存
downloadEntity.setCompletedSize(completedSize);
downloadDao.insertOrReplace(downloadEntity);
}
既然保存了,那么就要读取。这里读取分两种情况,在未退出Application的前提下读取,和重新进入Application的情况下读取。
其实这两种情况并没有什么区别,都是需要重新开启一个新的线程进行下载,所以在开启线程的时候读取数据库,看看有没有数据就可以了。
具体的代码就不写了。
4.结束