Android多线程下载示例详解
一、概述
说到android中的文件下载,android api中明确要求将耗时的操作放到一个子线程中执行,文件的下载无疑是需要耗费时间的,所以要将文件的下载放到子线程中执行。下面,我们一起来实现一个android中利用多线程下载文件的小例子。
二、服务端准备
在这个小例子中我以下载有道词典为例,在网上下载有道词典的安装包,在eclipse中新建项目web,将下载的有道词典安装包放置在webcontent目录下,并将项目发布到tomcat中,具体如下图所示
三、android实现
1、布局
界面上自上而下放置一个textview,用来提示文本框中输入的信息,一个文本框用来输入网络中下载文件的路径,一个button按钮,点击下载文件,一个progressbar显示下载进度,一个textview显示下载的百分比。具体布局内容如下:
<linearlayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" 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" tools:context=".mainactivity" > <textview android:layout_width="match_parent" android:layout_height="wrap_content" android:text="下载路径" /> <edittext android:id="@+id/ed_path" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="http://192.168.0.170:8080/web/youdao.exe"/> <button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="下载" android:onclick="download"/> <progressbar android:id="@+id/pb" android:layout_width="match_parent" android:layout_height="wrap_content" style="@android:style/widget.progressbar.horizontal"/> <textview android:id="@+id/tv_info" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:text="下载:0%"/> </linearlayout>
2、自定义progressbarlistener监听器接口
新建自定义progressbarlistener监听器接口,这个接口中定义两个方法,void getmax(int length)用来获取下载文件的长度,void getdownload(int length);用来获取每次下载的长度,这个方法中主要是在多线程中调用,子线程中获取到的数据传递到这两个接口方法中,然后在这两个接口方法中通过handler将相应的长度信息传递到主线程,更新界面显示信息,具体代码实现如下:
package com.example.inter; /** * 自定义进度条监听器 * @author liuyazhuang * */ public interface progressbarlistener { /** * 获取文件的长度 * @param length */ void getmax(int length); /** * 获取每次下载的长度 * @param length */ void getdownload(int length); }
3、自定义线程类downloadthread
这里通过继承thread的方式来实现自定义线程操作,在这个类中主要是实现文件的下载操作,在这个类中,定义了一系列与下载有关的实例变量来控制下载的数据,同时通过自定义监听器progressbarlistener中的void getdownload(int length)方法来跟新界面显示的进度信息。
具体实现如下:
package com.example.download; import java.io.file; import java.io.fileoutputstream; import java.io.inputstream; import java.io.randomaccessfile; import java.net.httpurlconnection; import java.net.url; import com.example.inter.progressbarlistener; /** * 自定义线程类 * @author liuyazhuang * */ public class downloadthread extends thread { //下载的线程id private int threadid; //下载的文件路径 private string path; //保存的文件 private file file; //下载的进度条更新的监听器 private progressbarlistener listener; //每条线程下载的数据量 private int block; //下载的开始位置 private int startposition; //下载的结束位置 private int endposition; public downloadthread(int threadid, string path, file file, progressbarlistener listener, int block) { this.threadid = threadid; this.path = path; this.file = file; this.listener = listener; this.block = block; this.startposition = threadid * block; this.endposition = (threadid + 1) * block - 1; } @override public void run() { super.run(); try { //创建randomaccessfile对象 randomaccessfile accessfile = new randomaccessfile(file, "rwd"); //跳转到开始位置 accessfile.seek(startposition); url url = new url(path); //打开http链接 httpurlconnection conn = (httpurlconnection) url.openconnection(); //设置超时时间 conn.setconnecttimeout(5000); //指定请求方式为get方式 conn.setrequestmethod("get"); //指定下载的位置 conn.setrequestproperty("range", "bytes="+startposition + "-" + endposition); //不用再去判断状态码是否为200 inputstream in = conn.getinputstream(); byte[] buffer = new byte[1024]; int len = 0; while((len = in.read(buffer)) != -1){ accessfile.write(buffer, 0, len); //更新下载进度 listener.getdownload(len); } accessfile.close(); in.close(); } catch (exception e) { // todo: handle exception e.printstacktrace(); } } }
4、新建downloadmanager类
这个类主要是对下载过程的管理,包括下载设置下载后文件要保存的位置,计算多线程中每个线程的数据下载量等等。
具体实现如下:
package com.example.download; import java.io.file; import java.io.randomaccessfile; import java.net.httpurlconnection; import java.net.url; import android.os.environment; import com.example.inter.progressbarlistener; /** * 文件下载管理器 * @author liuyazhuang * */ public class downloadmanager { //下载线程的数量 private static final int tread_size = 3; private file file; /** * 下载文件的方法 * @param path:下载文件的路径 * @param listener:自定义的下载文件监听接口 * @throws exception */ public void download(string path, progressbarlistener listener) throws exception{ url url = new url(path); httpurlconnection conn = (httpurlconnection) url.openconnection(); conn.setconnecttimeout(5000); conn.setrequestmethod("get"); if(conn.getresponsecode() == 200){ int filesize = conn.getcontentlength(); //设置进度条的最大长度 listener.getmax(filesize); //创建一个和服务器大小一样的文件 file = new file(environment.getexternalstoragedirectory(), this.getfilename(path)); randomaccessfile accessfile = new randomaccessfile(file, "rwd"); accessfile.setlength(filesize); //要关闭randomaccessfile对象 accessfile.close(); //计算出每条线程下载的数据量 int block = filesize % tread_size == 0 ? (filesize / tread_size) : (filesize / tread_size +1 ); //开启线程下载 for(int i = 0; i < tread_size; i++){ new downloadthread(i, path, file, listener, block).start(); } } } /** * 截取路径中的文件名称 * @param path:要截取文件名称的路径 * @return:截取到的文件名称 */ private string getfilename(string path){ return path.substring(path.lastindexof("/") + 1); } }
5、完善mainactivity
在这个类中首先,找到页面中的各个控件,实现button按钮的onclick事件,在onclick事件中开启一个线程进行下载操作,同时子线程中获取到的数据,通过handler与message机制传递到主线程,更新界面显示。
具体实现如下:
package com.example.multi; import android.app.activity; import android.os.bundle; import android.os.handler; import android.os.message; import android.view.menu; import android.view.view; import android.widget.edittext; import android.widget.progressbar; import android.widget.textview; import android.widget.toast; import com.example.download.downloadmanager; import com.example.inter.progressbarlistener; /** * mainactivity整个应用程序的入口 * @author liuyazhuang * */ public class mainactivity extends activity { protected static final int error_download = 0; protected static final int set_progress_max = 1; protected static final int update_progress = 2; private edittext ed_path; private progressbar pb; private textview tv_info; private downloadmanager manager; //handler操作 private handler mhandler = new handler(){ public void handlemessage(android.os.message msg) { switch (msg.what) { case error_download: //提示用户下载失败 toast.maketext(mainactivity.this, "下载失败", toast.length_short).show(); break; case set_progress_max: //得到最大值 int max = (integer) msg.obj; //设置进度条的最大值 pb.setmax(max); break; case update_progress: //获取当前下载的长度 int currentprogress = pb.getprogress(); //获取新下载的长度 int len = (integer) msg.obj; //计算当前总下载长度 int crrrenttotalprogress = currentprogress + len; pb.setprogress(crrrenttotalprogress); //获取总大小 int maxprogress = pb.getmax(); //计算百分比 float value = (float)currentprogress / (float)maxprogress; int percent = (int) (value * 100); //显示下载的百分比 tv_info.settext("下载:"+percent+"%"); break; default: break; } }; }; @override protected void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.activity_main); this.ed_path = (edittext) super.findviewbyid(r.id.ed_path); this.pb = (progressbar) super.findviewbyid(r.id.pb); this.tv_info = (textview) super.findviewbyid(r.id.tv_info); this.manager = new downloadmanager(); } @override public boolean oncreateoptionsmenu(menu menu) { // inflate the menu; this adds items to the action bar if it is present. getmenuinflater().inflate(r.menu.main, menu); return true; } public void download(view v){ final string path = ed_path.gettext().tostring(); //下载 new thread(new runnable() { @override public void run() { // todo auto-generated method stub try { manager.download(path, new progressbarlistener() { @override public void getmax(int length) { // todo auto-generated method stub message message = new message(); message.what = set_progress_max; message.obj = length; mhandler.sendmessage(message); } @override public void getdownload(int length) { // todo auto-generated method stub message message = new message(); message.what = update_progress; message.obj = length; mhandler.sendmessage(message); } }); } catch (exception e) { // todo: handle exception e.printstacktrace(); message message = new message(); message.what = error_download; mhandler.sendmessage(message); } } }).start(); } }
6、增加权限
最后,别忘了给应用授权,这里要用到android联网授权和向sd卡中写入文件的权限。
具体实现如下:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.multi" android:versioncode="1" android:versionname="1.0" > <uses-sdk android:minsdkversion="8" android:targetsdkversion="18" /> <uses-permission android:name="android.permission.internet"/> <uses-permission android:name="android.permission.mount_unmount_filesystems"/> <uses-permission android:name="android.permission.write_external_storage"/> <application android:allowbackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/apptheme" > <activity android:name="com.example.multi.mainactivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.main" /> <category android:name="android.intent.category.launcher" /> </intent-filter> </activity> </application> </manifest>
四、运行效果
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。