欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android多线程下载示例详解

程序员文章站 2023-12-03 10:55:22
一、概述          说到android中的文件下载,android api中明...

一、概述

         说到android中的文件下载,android api中明确要求将耗时的操作放到一个子线程中执行,文件的下载无疑是需要耗费时间的,所以要将文件的下载放到子线程中执行。下面,我们一起来实现一个android中利用多线程下载文件的小例子。

二、服务端准备

        在这个小例子中我以下载有道词典为例,在网上下载有道词典的安装包,在eclipse中新建项目web,将下载的有道词典安装包放置在webcontent目录下,并将项目发布到tomcat中,具体如下图所示

Android多线程下载示例详解

三、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> 

四、运行效果

Android多线程下载示例详解

Android多线程下载示例详解

Android多线程下载示例详解

提醒:大家可以到这个链接来获取完整的代码示例

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。