原文出处 : Android开发——断点续传原理以及实现
原文作者 : SEU_Calvin
感谢原文作者无私的分享 !
1. 断点续传原理
在本地下载过程中要使用数据库实时存储到底存储到文件的哪个位置了,这样点击开始继续传递时,才能通过HTTP的GET请求中的setRequestProperty()方法可以告诉服务器,数据从哪里开始,到哪里结束。同时在本地的文件写入时,RandomAccessFile的seek()方法也支持在文件中的任意位置进行写入操作。同时通过广播将子线程的进度告诉Activity的ProcessBar。
2. Activity的按钮响应
当点击开始按钮时,将url写在了FileInfo类的对象info中并通过Intent从Activity传递到了Service中。这里使用setAction()来区分是开始按钮还是暂停按钮。
[java] view plain copy
- public class FileInfo implements Serializable{
- private String url;
- private int length;
- private int start;
- private int now;
-
- }
-
- strat.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View view) {
- Intent intent = new Intent(MainActivity.this,DownLoadService.class);
- intent.setAction(DownLoadService.ACTION_START);
- intent.putExtra("fileUrl",info);
- startService(intent);
- }
- });
3. 在Service中的子线程中获取文件大小
在Service中的onStartCommand()中,将FileInfo对象从Intent中取出,如果是开始命令,则开启一个线程,根据该url去获得要下载文件的大小,将该大小写入对象并通过Handler传回Service,同时在本地创建一个相同大小的本地文件。暂停命令最后会讲到。
[java] view plain copy
- public void run() {
- HttpURLConnection urlConnection = null;
- RandomAccessFile randomFile = null;
- try {
- URL url = new URL(fileInfo.getUrl());
- urlConnection = (HttpURLConnection) url.openConnection();
- urlConnection.setConnectTimeout(3000);
- urlConnection.setRequestMethod("GET");
- int length = -1;
- if (urlConnection.getResponseCode() == HttpStatus.SC_OK) {
-
- length = urlConnection.getContentLength();
- }
- if (length <= 0) {
- return;
- }
-
- File dir = new File(DOWNLOAD_PATH);
- if (!dir.exists()) {
- dir.mkdir();
- }
- File file = new File(dir, FILE_NAME);
- randomFile = new RandomAccessFile(file, "rwd");
- randomFile.setLength(length);
-
- fileInfo.setLength(length);
-
- mHandle.obtainMessage(0, fileInfo).sendToTarget();
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- }
- }
- }
4. 数据库操作封装
在Service的handleMessage()方法中拿到有length属性的FileInfo对象,并使用自定义的DownLoadUtil类进行具体的文件下载逻辑。这里传入上下文,因为数据库处理操作需要用到。
[java] view plain copy
- downLoadUtil = new DownLoadUtil(DownLoadService.this,info);
- downLoadUtil.download();
这里有一个数据库操作的接口ThreadDAO,内部有增删改查等逻辑,用于记录下载任务的信息。自定义一个ThreadDAOImpl类将这里的逻辑实现,内部数据库创建关于继承SQLiteOpenHelper的自定义类的逻辑就不贴了,比较简单,该类会在ThreadDAOImpl类的构造方法中创建实例。完成底层数据库操作的封装。
[java] view plain copy
- public interface ThreadDAO {
-
- public void insert(FileInfo info);
-
- public void delete(String url);
-
- public void update(String url,int finished);
-
- public List<FileInfo> get(String url);
-
- public boolean isExits(String url);
- }
5. 具体的文件下载逻辑
[java] view plain copy
- public class DownLoadUtil {
-
- public void download(){
- List<FileInfo> lists = threadDAO.get(fileInfo.getUrl());
- FileInfo info = null;
- if(lists.size() == 0){
-
- new MyThread(fileInfo).start();
- }else{
-
- info = lists.get(0);
- new MyThread(info).start();
- }
- }
-
- class MyThread extends Thread{
- private FileInfo info = null;
- public MyThread(FileInfo threadInfo) {
- this.info = threadInfo;
- }
- @Override
- public void run() {
-
- if(!threadDAO.isExits(info.getUrl())){
- threadDAO.insert(info);
- }
- HttpURLConnection urlConnection = null;
- RandomAccessFile randomFile =null;
- InputStream inputStream = null;
- try {
- URL url = new URL(info.getUrl());
- urlConnection = (HttpURLConnection) url.openConnection();
- urlConnection.setConnectTimeout(3000);
- urlConnection.setRequestMethod("GET");
-
- int start = info.getStart() + info.getNow();
- urlConnection.setRequestProperty("Range","bytes=" + start + "-" + info.getLength());
-
-
- File file = new File(DOWNLOAD_PATH,FILE_NAME);
- randomFile = new RandomAccessFile(file, "rwd");
- randomFile.seek(start);
-
-
- Intent intent = new Intent(ACTION_UPDATE);
- finished += info.getNow();
-
- if (urlConnection.getResponseCode() == HttpStatus.SC_PARTIAL_CONTENT) {
-
- inputStream = urlConnection.getInputStream();
- byte[] buffer = new byte[512];
- int len = -1;
- long time = System.currentTimeMillis();
- while ((len = inputStream.read(buffer))!= -1){
-
- randomFile.write(buffer,0,len);
-
- finished += len;
-
- if(System.currentTimeMillis() - time >500){
- time = System.currentTimeMillis();
- intent.putExtra("now",finished *100 /fileInfo.getLength());
- context.sendBroadcast(intent);
- }
-
- if(isPause){
- threadDAO.update(info.getUrl(),finished);
- return;
- }
- }
-
- threadDAO.delete(info.getUrl());
- }
- }catch (Exception e){
- e.printStackTrace();
- }finally {
- }
- }
- }
- }
上面也讲到使用自定义的DownLoadUtil类进行具体的文件下载逻辑,这也是最关键的部分了,在该类的构造方法中进行ThreadDAOImpl实例的创建。并在download()中通过数据库查询的操作,判断是否是第一次开始下载任务,如果是,则开启一个子线程MyThread进行下载任务,否则将进度信息从数据库中取出,并将该信息传递给MyThread。
在MyThread中,通过info.getStart() + info.getNow()设置开始下载的位置,如果是第一次下载两个数将都是0,如果是暂停后再下载,则info.getNow()会取出非0值,该值来自数据库存储。使用setRequestProperty告知服务器从哪里开始传递数据,传递到哪里结束,本地使用RandomAccessFile的seek()方法进行数据的本地存储。使用广播将进度的百分比传递给Activity,Activity再改变ProcessBar进行UI调整。
这里很关键的一点是在用户点击暂停后会在Service中调用downLoadUtil.isPause = true,因此上面while循环会结束,停止下载并通过数据库的update()保存进度值。从而在续传时取出该值,重新对服务器发起文件起始点的下载任务请求,同时也在本地文件的相应位置继续写入操作。
6. 效果如下所示