刚刚有人问我断点续传的原理是什么?没有特别完整的说出来,实在是有点羞愧,自己理解的就是利用数据库在本地写缓存,然后把那个暂停的点记录一下,然后下一次再从那个点开始请求数据,现在一般都是用第三方的,导致具体的怎么实现的原理都有点忘记了~哎~真是对不起自己啊


原理 :

  • 从字面上理解,所谓断点续传就是从停止的地方重新下载。 断点:线程停止的位置。 续传:从停止的位置重新下载。
  • 用代码解析就是:断点: 当前线程已经下载完成的数据长度。续传: 向服务器请求上次线程停止位置之后的数据。原理知道了,功能实现起来也简单。每当线程停止时就把已下载的数据长度写入记录文件,当重新下载时,从记录文件读取已经下载了的长度。而这个长度就是所需要的断点。
  • 总结来说就是下载过程中要使用数据库实时存储到底存储到文件的哪个位置了,这样点击开始继续传递时,才能通过HTTP的GET请求中的setRequestProperty()方法可以告诉服务器,数据从哪里开始,到哪里结束。同时在本地的文件写入时,RandomAccessFile的seek()方法也支持在文件中的任意位置进行写入操作。同时通过广播将子线程的进度告诉Activity的ProcessBar。
     

代码示意 :

Activity的按钮响应

当点击开始按钮时,将url写在了FileInfo类的对象info中并通过Intent从Activity传递到了Service中。这里使用setAction()来区分是开始按钮还是暂停按钮。

public class FileInfo implements Serializable{
    private String url; //URL
    private int length; //长度或结束位置
    private int start; //开始位置
    private int now;//当前进度
//构造方法,set/get略
}
//开始按钮逻辑,停止逻辑大致相同
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);
   }
});

在Service中的子线程中获取文件大小

在Service中的onStartCommand()中,将FileInfo对象从Intent中取出,如果是开始命令,则开启一个线程,根据该url去获得要下载文件的大小,将该大小写入对象并通过Handler传回Service,同时在本地创建一个相同大小的本地文件。暂停命令最后会讲到。

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对象
                fileInfo.setLength(length);
                //通过Handler将对象传递给Service
                mHandle.obtainMessage(0, fileInfo).sendToTarget();
            } catch (Exception e) {
                e.printStackTrace();
            } finally {  //流的回收逻辑略
            }
        }
    }

数据库操作封装

在Service的handleMessage()方法中拿到有length属性的FileInfo对象,并使用自定义的DownLoadUtil类进行具体的文件下载逻辑。这里传入上下文,因为数据库处理操作需要用到。

downLoadUtil = new DownLoadUtil(DownLoadService.this,info);
downLoadUtil.download();

这里有一个数据库操作的接口ThreadDAO,内部有增删改查等逻辑,用于记录下载任务的信息。自定义一个ThreadDAOImpl类将这里的逻辑实现,内部数据库创建关于继承SQLiteOpenHelper的自定义类的逻辑就不贴了,比较简单,该类会在ThreadDAOImpl类的构造方法中创建实例。完成底层数据库操作的封装。

public interface ThreadDAO {
    //插入一条数据
    public void insert(FileInfo info);
    //根据URL删除一条数据
    public void delete(String url);
    //根据URL更新一条进度
    public void update(String url,int finished);
    //根据URL找到一条数据
    public List<FileInfo> get(String url);
    //是否存在
    public boolean isExits(String url);
}

具体的文件下载逻辑

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);
 
                //向Activity发广播
                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);
                        //把进度发送给Activity
                        finished += len;
                        //看时间间隔,时间间隔大于500ms再发
                        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()保存进度值。从而在续传时取出该值,重新对服务器发起文件起始点的下载任务请求,同时也在本地文件的相应位置继续写入操作。

最后补充一点,关于断线续传的HTTP状态码并不是200,而是206,即HttpStatus.SC_PARTIAL_CONTENT。


效果图: