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>

四、运行效果

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

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

(0)

相关推荐

  • android中多线程下载实例

    复制代码 代码如下: public class MainActivity extends Activity { // 声明控件 // 路径与线程数量 private EditText et_url, et_num; // 进度条 public static ProgressBar pb_thread; // 显示进度的操作 private TextView tv_pb; // 线程的数量 public static int threadNum = 3; // 每个线程负责下载的大小 public

  • Android实现多线程下载文件的方法

    本文实例讲述了Android实现多线程下载文件的方法.分享给大家供大家参考.具体如下: 多线程下载大概思路就是通过Range 属性实现文件分段,然后用RandomAccessFile 来读写文件,最终合并为一个文件 首先看下效果图: 创建工程 ThreadDemo 首先布局文件 threaddemo.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=&quo

  • Android编程开发实现多线程断点续传下载器实例

    本文实例讲述了Android编程开发实现多线程断点续传下载器.分享给大家供大家参考,具体如下: 使用多线程断点续传下载器在下载的时候多个线程并发可以占用服务器端更多资源,从而加快下载速度,在下载过程中记录每个线程已拷贝数据的数量,如果下载中断,比如无信号断线.电量不足等情况下,这就需要使用到断点续传功能,下次启动时从记录位置继续下载,可避免重复部分的下载.这里采用数据库来记录下载的进度. 效果图:   断点续传 1.断点续传需要在下载过程中记录每条线程的下载进度 2.每次下载开始之前先读取数据库

  • Android版多线程下载 仿下载助手(最新)

    首先声明一点: 这里的多线程下载并不是指多个线程下载一个 文件,而是每个线程负责一个文件,今天给大家分享一个多线程下载的 例子.先看一下效果,点击下载开始下载,同时显示下载进度,下载完成,变成程安装,点击安装提示安装应用. 界面效果图: 线程池ThreadPoolExecutor ,先简单学习下这个线程池的使用 /** * Parameters: corePoolSize the number of threads to keep in the pool, even if they are id

  • Android编程开发实现带进度条和百分比的多线程下载

    本文实例讲述了Android编程开发实现带进度条和百分比的多线程下载.分享给大家供大家参考,具体如下: 继上一篇<java多线程下载实例详解>之后,可以将它移植到我们的安卓中来,下面是具体实现源码: DownActivity.java: package com.example.downloads; import java.io.File; import java.io.IOException; import java.io.RandomAccessFile; import java.net.H

  • Android实现多线程断点下载的方法

    本文实例讲述了Android实现多线程断点下载的方法.分享给大家供大家参考.具体实现方法如下: package cn.itcast.download; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputSt

  • Android实现多线程下载图片的方法

    很多时候我们需要在Android设备上下载远程服务器上的图片进行显示,今天整理出两种比较好的方法来实现远程图片的下载. 方法一.直接通过Android提供的Http类访问远程服务器,这里AndroidHttpClient是SDK 2.2中新出的方法,API Level为8,大家需要注意下,静态访问可以直接调用,如果SDK版本较低可以考虑Apache的Http库,当然HttpURLConnection 或URLConnection也可以. static Bitmap downloadBitmapB

  • Android实现网络多线程断点续传下载实例

    我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多线程断点需要什么功能? 1.多线程下载, 2.支持断点. 使用多线程的好处:使用多线程下载会提升文件下载的速度.那么多线程下载文件的过程是: (1)首先获得下载文件的长度,然后设置本地文件的长度. HttpURLConnection.getContentLength();//获取下载文件的长度 RandomAccessFile file = new RandomAc

  • Android实现网络多线程文件下载

    实现原理 (1)首先获得下载文件的长度,然后设置本地文件的长度. (2)根据文件长度和线程数计算每条线程下载的数据长度和下载位置. 如:文件的长度为6M,线程数为3,那么,每条线程下载的数据长度为2M,每条线程开始下载的位置如下图所示: (网上找的图) 例如10M大小,使用3个线程来下载, 线程下载的数据长度 (10%3 == 0 ? 10/3:10/3+1) ,第1,2个线程下载长度是4M,第三个线程下载长度为2M 下载开始位置:线程id*每条线程下载的数据长度 = ? 下载结束位置:(线程i

  • android实现多线程下载文件(支持暂停、取消、断点续传)

    多线程下载文件(支持暂停.取消.断点续传) 多线程同时下载文件即:在同一时间内通过多个线程对同一个请求地址发起多个请求,将需要下载的数据分割成多个部分,同时下载,每个线程只负责下载其中的一部分,最后将每一个线程下载的部分组装起来即可. 涉及的知识及问题 请求的数据如何分段 分段完成后如何下载和下载完成后如何组装到一起 暂停下载和继续下载的实现(wait().notifyAll().synchronized的使用) 取消下载和断点续传的实现 一.请求的数据如何分段 首先通过HttpURLConne

随机推荐