Android多文件断点续传在很多应用场景中都会运用到,更重要的是相对于简单的下载功能,断点续传在下载文件过程中能带来非常好的用户体验。本系列教程将围绕一个简单Demo介绍多文件断点续传的实现方式。

先看效果图,源码在教程结尾提供。

Demo所涉及主要内容如下:

1. Service:用于后台处理下载文件的逻辑。

2. SQLite : 用于保存下载进度。

3. EventBus : 用于分发和接收下载进度。

4. ThreadPool : 用于管理下载线程。

一. 封装实体类

我们需要将下载的文件信息和下载线程的信息分别封装起来。

/** * Created by kun on 2016/11/10. * 下载文件信息 */public class FileBean implements Serializable {    private int id;    private String fileName;    private String url;    private int length;    private int finished;    .... //Constructor,get,set}
/** * Created by kun on 2016/11/10. * 下载线程信息 */public class ThreadBean implements Serializable{    private int id;    private String url;    private int start;    private int end;    private int finished;    .... //Constructor,get,set}

FileBean 中封装了下载文件的信息:id、下载路径、文件名称、文件长度和已下载的长度。

ThreadBean 中封装了下载线程的信息:id、下载路径、下载起始位置、下载结束位置和已下载的长度。

二.绘制布局以及添加数据

在效果图中我们看到界面很简单,这里用RecyclerView来实现。

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:orientation="vertical">    <android.support.v7.widget.RecyclerView  android:id="@+id/recyclerview" android:layout_width="match_parent" android:layout_height="match_parent"/></LinearLayout>

在Activity中我们初始化RecyclerView并添加几个下载数据

    private void initView(){        recyclerview = (RecyclerView) findViewById(R.id.recyclerview);        LinearLayoutManager layoutManager = new LinearLayoutManager(this);        layoutManager.setOrientation(LinearLayoutManager.VERTICAL);        recyclerview.setLayoutManager(layoutManager);    }    private void initData() {        FileBean fileBean1 = new FileBean(0, "instmobilemgr.exe", url1, 0);        FileBean fileBean2 = new FileBean(1, "QQDownload_Setup_48_773_400.exe", url2, 0);        FileBean fileBean3 = new FileBean(2, "QQPlayer_Setup_39_936.exe", url3, 0);        FileBean fileBean4 = new FileBean(3, "QQMusicForYQQ.exe", url4, 0);        List<FileBean> fileBeanList = new ArrayList<>();        fileBeanList.add(fileBean1);        fileBeanList.add(fileBean2);        fileBeanList.add(fileBean3);        fileBeanList.add(fileBean4);        adaper = new RecyclerViewListAdapter(this, fileBeanList);        recyclerview.setAdapter(adaper);    }

三.RecyclerViewListAdapter

/** * Created by kun on 2016/11/11. */public class RecyclerViewListAdapter extends RecyclerView.Adapter<RecyclerViewListAdapter.ViewHolder> {    List<FileBean> datas;    Context context;    public RecyclerViewListAdapter(Context context, List<FileBean> datas) {        if (datas == null) datas = new ArrayList<>();        this.datas = datas;        this.context = context;    }    ... ...    //自定义的ViewHolder,持有每个Item的的所有界面元素    public static class ViewHolder extends RecyclerView.ViewHolder {        TextView textName;        ProgressBar progressBar;        Button btnStart;        Button btnPause;        public ViewHolder(View convertView) {            super(convertView);            textName = (TextView) convertView.findViewById(R.id.textName);            progressBar = (ProgressBar) convertView.findViewById(R.id.progressBar);            btnStart = (Button) convertView.findViewById(R.id.btnStart);            btnPause = (Button) convertView.findViewById(R.id.btnPause);        }    }    //将数据与界面进行绑定的操作    @Override    public void onBindViewHolder(final ViewHolder viewHoder, final int position) {        final FileBean fileBean = datas.get(position);        viewHoder.textName.setText(fileBean.getFileName());        if(fileBean.getLength()!=0) {            viewHoder.progressBar.setProgress((int)(fileBean.getFinished()*1.0f/fileBean.getLength()*100));        }        viewHoder.btnStart.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                Intent startIntent = new Intent(context, DownloadService.class);                startIntent.setAction(DownloadService.ACTION_START);                startIntent.putExtra("FileBean", fileBean);                context.startService(startIntent);            }        });        viewHoder.btnPause.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                Intent pauseIntent = new Intent(context, DownloadService.class);                pauseIntent.setAction(DownloadService.ACTION_PAUSE);                pauseIntent.putExtra("FileBean", fileBean);                context.startService(pauseIntent);            }        });    }    private long curTime = 0;     public void updateProgress(FileBean fileBean) {         int i = 0;        for(FileBean data:datas){            if(data.getId() == fileBean.getId()){                data.setLength(fileBean.getLength());                data.setFinished(fileBean.getFinished());                if(System.currentTimeMillis()-curTime >500) {                    curTime = System.currentTimeMillis();                    notifyDataSetChanged();                }                return;            }            i++;        }    }}

在这里我只贴出了关键代码,完整代码请查看源码。这里需要注意的是定义了一个公共的方法:updateProgress()。用于外部调用刷新进度条,这里传入了一个FileBean,通过Id对比我们可以获取到对应的数据,将文件长度和文件已下载完成的进度赋值过去,然后通过notifyDataSetChanged方法通知UI更新。这里对刷新做了限制,最快为500毫秒刷新一次。

刷新列表本来是采用notifyItemChanged(int position)方法,可惜会出现闪烁现象,找不到比较合理的解决方案,希望在此能抛砖引玉。

接着我们看一下onBindViewHolder方法中开始按钮和暂停按钮的点击事件。可以看到是这里主要启动了一个DownloadService,将对应的FileBean和Action传递到Service中,接着由Service在后台处理下载的逻辑。

到这里界面基本就处理完了,在进入DownloadService处理前我们还需要先准备好数据库,欢迎继续阅读下一篇。

Android多文件断点续传(二)——实现数据库储存下载信息

更多相关文章

  1. 一款常用的 Squid 日志分析工具
  2. GitHub 标星 8K+!一款开源替代 ls 的工具你值得拥有!
  3. RHEL 6 下 DHCP+TFTP+FTP+PXE+Kickstart 实现无人值守安装
  4. Linux 环境下实战 Rsync 备份工具及配置 rsync+inotify 实时同步
  5. Android本地文件管理器思路解析一一增删改查具体实现
  6. 36、Android编写应用-为新设备添加模块
  7. Android的代码规范及阿里巴巴最新Java开发手册福利
  8. Android(安卓)Handler解析
  9. 面试经典题Handler机制回答

随机推荐

  1. html之盒模型,及字体图标的使用
  2. 运营级在线客服/来客客服/带非常完整的搭
  3. 图标以及盒模型
  4. 固定定位、三行三列布局
  5. 0701作业
  6. 实战:简单的注册页面与常用选择器
  7. webgl反向入门,从NDC坐标到屏幕坐标
  8. 字体图标用法与盒模型页面布局
  9. PHP Redis函数查询
  10. 盒模型与字体图标使用