一、概述
说到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" /> <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,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>
四、运行效果
提醒:大家可以到这个链接来获取完整的代码示例。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。
总结以上是内存溢出为你收集整理的Android多线程下载示例详解全部内容,希望文章能够帮你解决Android多线程下载示例详解所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)