本文续接我上一篇文章《Android实战:简易断点续传下载器实现》
链接地址:http://www.jianshu.com/p/5b2e22c42467
说到多线程下载,也许大家会觉得很迷惑,但多线程的原理实际上与单线程下载的原理并无区别。
多线程下载只需要确定好下载一个文件需要多少个线程,一般来说最好为3条线程,因为线程过多会占用系统资源,而且线程间的相互竞争也会导致下载变慢。
其次下载的时候将文件分割为三份(假设用3条线程下载)下载,在java中就要用到上次提到的RandomAccessFile这个API,它的开始结束为止用以下代码确定:
conn.setRequestProperty("Range", "bytes=" + start + "-" + end)
最后就是断点续传了,只需要才程序停止下载的时候记录下最后的下载位置就好了,当下次下载的时候从当前停止的位置开始下载。
OK,那么现在就开始我们的多线程下载+通知栏控制的实战之旅吧!
多线程断点续传下载
我们这次要做的并非简单的多线程下载,而是要做到多文件多线程的同时下载
重写布局
这次下载需要展示多个下载的文件,所以使用ListView控件,界面效果如下
下载界面.png
每个ListView的item都很简单,基本上只需要将上次写的下载界面搬过来就好了。
新建一个Layout,命名为item,将中的界面布局剪切过来,然后在中设置一个ListView空间。
activity_main.xml代码如下
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
android:id="@+id/list_view"
android:layout_width="match_parent"
android:layout_height="match_parent" />
至于Item的布局,为了省功夫,就不写了,大家可以去我的Github下载名为MultiDownload的项目来参考。
建立FileAdapter类
布局写好了,但是ListView总是要有个Adapter类来绑定视图,填充布局的不是么,所以接下来就开始写FileAdapter了。
话说回来,ListView真的是个很重要的空间,不熟悉的小伙伴抓紧多看看怎么做吧。
创建一个继承自BaseAdapter类的FileAdapter,里面拥有以下三个成员变量:
private Context mContext = null;
private List mFilelist = null;
private LayoutInflater layoutInflater;
然后重写构造函数:
public FileAdapter(Context mContext, List mFilelist) {
this.mContext = mContext;
this.mFilelist = mFilelist;
layoutInflater = LayoutInflater.from(mContext);
}
再将继承的getCount/getItem/getItemId三个方法的返回值写好,用于ListView找到各自的Item。
接下来就是重头戏,重写getView方法了!
我们先定义一个静态的ViewHolder内部类,这样在ListView属性的时候才不会重复创建对象,减轻内存压力,这个谷歌官方推荐的哦!
static class ViewHolder {
TextView textview;
Button startButton;
Button stopButton;
ProgressBar progressBar;
}
然后在getView中绑定布局item中的各个控件,并且设置按钮的点击事件,getView代码如下:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder = null;
final FileInfo mFileInfo = mFilelist.get(position);
if (convertView == null) {
convertView = layoutInflater.inflate(R.layout.item, null);
viewHolder = new ViewHolder();
viewHolder.textview = (TextView) convertView.findViewById(R.id.file_textview);
viewHolder.startButton = (Button) convertView.findViewById(R.id.start_button);
viewHolder.stopButton = (Button) convertView.findViewById(R.id.stop_button);
viewHolder.progressBar = (ProgressBar) convertView.findViewById(R.id.progressBar2);
viewHolder.textview.setText(mFileInfo.getFileName());
viewHolder.progressBar.setMax(100);
viewHolder.startButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(mContext, DownloadService.class); intent.setAction(DownloadService.ACTION_START);
intent.putExtra("fileInfo", mFileInfo);
mContext.startService(intent);
}
});
viewHolder.stopButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(mContext, DownloadService.class);
intent.setAction(DownloadService.ACTION_STOP);
intent.putExtra("fileInfo", mFileInfo);
mContext.startService(intent);
}
});
convertView.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) convertView.getTag();
}
viewHolder.progressBar.setProgress(mFileInfo.getFinished());
return convertView;
}
最后再新建一个更新进度条的方法,在获得文件ID,和当前进度之后,直接更新进度条,代码如下:
public void updataProgress(int id, int progress) { <