实现原理

(1)首先获得下载文件的长度,然后设置本地文件的长度。

(2)根据文件长度和线程数计算每条线程下载的数据长度和下载位置。

如:文件的长度为6M,线程数为3,那么,每条线程下载的数据长度为2M,每条线程开始下载的位置如下图所示:

(网上找的图)

例如10M大小,使用3个线程来下载,

线程下载的数据长度   (10%3 == 0 ? 10/3:10/3+1) ,第1,2个线程下载长度是4M,第三个线程下载长度为2M
下载开始位置:线程id*每条线程下载的数据长度 = ?
下载结束位置:(线程id+1)*每条线程下载的数据长度-1=?

之前练习时的一个demo,不多说了,直接上代码吧,有关断点续传,需要使用数据库,不再加了,网上有很多成熟的项目可以直接用。

实例

MainApp:

  1. package com.amos.app;
  2.  
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.net.MalformedURLException;
  6. import java.net.URL;
  7. import java.net.URLConnection;
  8. import com.amos.download.R;
  9. import android.annotation.SuppressLint;
  10. import android.app.Activity;
  11. import android.os.Bundle;
  12. import android.os.Environment;
  13. import android.os.Handler;
  14. import android.os.Message;
  15. import android.util.Log;
  16. import android.view.View;
  17. import android.view.View.OnClickListener;
  18. import android.widget.ProgressBar;
  19. import android.widget.TextView;
  20. import android.widget.Toast;
  21.  
  22. /**
  23. * @author yangxiaolong
  24. * @2014-5-6
  25. */
  26. public class MainApp extends Activity implements OnClickListener {
  27.  
  28. private static final String TAG = MainApp.class.getSimpleName();
  29.  
  30. /** 显示下载进度TextView */
  31. private TextView mMessageView;
  32. /** 显示下载进度ProgressBar */
  33. private ProgressBar mProgressbar;
  34.  
  35. @Override
  36. protected void onCreate(Bundle savedInstanceState) {
  37. super.onCreate(savedInstanceState);
  38. setContentView(R.layout.progress_activity);
  39. findViewById(R.id.download_btn).setOnClickListener(this);
  40. mMessageView = (TextView) findViewById(R.id.download_message);
  41. mProgressbar = (ProgressBar) findViewById(R.id.download_progress);
  42. }
  43.  
  44. @Override
  45. public void onClick(View v) {
  46. if (v.getId() == R.id.download_btn) {
  47. doDownload();
  48. }
  49. }
  50.  
  51. /**
  52. * 使用Handler更新UI界面信息
  53. */
  54. @SuppressLint("HandlerLeak")
  55. Handler mHandler = new Handler() {
  56. @Override
  57. public void handleMessage(Message msg) {
  58.  
  59. mProgressbar.setProgress(msg.getData().getInt("size"));
  60.  
  61. float temp = (float) mProgressbar.getProgress()
  62. / (float) mProgressbar.getMax();
  63.  
  64. int progress = (int) (temp * 100);
  65. if (progress == 100) {
  66. Toast.makeText(MainApp.this, "下载完成!", Toast.LENGTH_LONG).show();
  67. }
  68. mMessageView.setText("下载进度:" + progress + " %");
  69.  
  70. }
  71. };
  72.  
  73. /**
  74. * 下载准备工作,获取SD卡路径、开启线程
  75. */
  76. private void doDownload() {
  77. // 获取SD卡路径
  78. String path = Environment.getExternalStorageDirectory()
  79. + "/amosdownload/";
  80. File file = new File(path);
  81. // 如果SD卡目录不存在创建
  82. if (!file.exists()) {
  83. file.mkdir();
  84. }
  85. // 设置progressBar初始化
  86. mProgressbar.setProgress(0);
  87.  
  88. // 简单起见,我先把URL和文件名称写死,其实这些都可以通过HttpHeader获取到
  89. String downloadUrl = "http://gdown.baidu.com/data/wisegame/91319a5a1dfae322/baidu_16785426.apk";
  90. String fileName = "baidu_16785426.apk";
  91. int threadNum = 5;
  92. String filepath = path + fileName;
  93. Log.d(TAG, "download file path:" + filepath);
  94. downloadTask task = new downloadTask(downloadUrl, threadNum, filepath);
  95. task.start();
  96. }
  97.  
  98. /**
  99. * 多线程文件下载
  100. *
  101. * @author yangxiaolong
  102. * @2014-8-7
  103. */
  104. class downloadTask extends Thread {
  105. private String downloadUrl;// 下载链接地址
  106. private int threadNum;// 开启的线程数
  107. private String filePath;// 保存文件路径地址
  108. private int blockSize;// 每一个线程的下载量
  109.  
  110. public downloadTask(String downloadUrl, int threadNum, String fileptah) {
  111. this.downloadUrl = downloadUrl;
  112. this.threadNum = threadNum;
  113. this.filePath = fileptah;
  114. }
  115.  
  116. @Override
  117. public void run() {
  118.  
  119. FileDownloadThread[] threads = new FileDownloadThread[threadNum];
  120. try {
  121. URL url = new URL(downloadUrl);
  122. Log.d(TAG, "download file http path:" + downloadUrl);
  123. URLConnection conn = url.openConnection();
  124. // 读取下载文件总大小
  125. int fileSize = conn.getContentLength();
  126. if (fileSize <= 0) {
  127. System.out.println("读取文件失败");
  128. return;
  129. }
  130. // 设置ProgressBar最大的长度为文件Size
  131. mProgressbar.setMax(fileSize);
  132.  
  133. // 计算每条线程下载的数据长度
  134. blockSize = (fileSize % threadNum) == 0 ? fileSize / threadNum
  135. : fileSize / threadNum + 1;
  136.  
  137. Log.d(TAG, "fileSize:" + fileSize + " blockSize:");
  138.  
  139. File file = new File(filePath);
  140. for (int i = 0; i < threads.length; i++) {
  141. // 启动线程,分别下载每个线程需要下载的部分
  142. threads[i] = new FileDownloadThread(url, file, blockSize,
  143. (i + 1));
  144. threads[i].setName("Thread:" + i);
  145. threads[i].start();
  146. }
  147.  
  148. boolean isfinished = false;
  149. int downloadedAllSize = 0;
  150. while (!isfinished) {
  151. isfinished = true;
  152. // 当前所有线程下载总量
  153. downloadedAllSize = 0;
  154. for (int i = 0; i < threads.length; i++) {
  155. downloadedAllSize += threads[i].getDownloadLength();
  156. if (!threads[i].isCompleted()) {
  157. isfinished = false;
  158. }
  159. }
  160. // 通知handler去更新视图组件
  161. Message msg = new Message();
  162. msg.getData().putInt("size", downloadedAllSize);
  163. mHandler.sendMessage(msg);
  164. // Log.d(TAG, "current downloadSize:" + downloadedAllSize);
  165. Thread.sleep(1000);// 休息1秒后再读取下载进度
  166. }
  167. Log.d(TAG, " all of downloadSize:" + downloadedAllSize);
  168.  
  169. } catch (MalformedURLException e) {
  170. e.printStackTrace();
  171. } catch (IOException e) {
  172. e.printStackTrace();
  173. } catch (InterruptedException e) {
  174. e.printStackTrace();
  175. }
  176.  
  177. }
  178. }
  179.  
  180. }

FileDownloadThread:

  1. package com.amos.app;
  2.  
  3. import java.io.BufferedInputStream;
  4. import java.io.File;
  5. import java.io.IOException;
  6. import java.io.RandomAccessFile;
  7. import java.net.URL;
  8. import java.net.URLConnection;
  9. import android.util.Log;
  10.  
  11. /**
  12. * 文件下载类
  13. *
  14. * @author yangxiaolong
  15. * @2014-5-6
  16. */
  17. public class FileDownloadThread extends Thread {
  18.  
  19. private static final String TAG = FileDownloadThread.class.getSimpleName();
  20.  
  21. /** 当前下载是否完成 */
  22. private boolean isCompleted = false;
  23. /** 当前下载文件长度 */
  24. private int downloadLength = 0;
  25. /** 文件保存路径 */
  26. private File file;
  27. /** 文件下载路径 */
  28. private URL downloadUrl;
  29. /** 当前下载线程ID */
  30. private int threadId;
  31. /** 线程下载数据长度 */
  32. private int blockSize;
  33.  
  34. /**
  35. *
  36. * @param url:文件下载地址
  37. * @param file:文件保存路径
  38. * @param blocksize:下载数据长度
  39. * @param threadId:线程ID
  40. */
  41. public FileDownloadThread(URL downloadUrl, File file, int blocksize,
  42. int threadId) {
  43. this.downloadUrl = downloadUrl;
  44. this.file = file;
  45. this.threadId = threadId;
  46. this.blockSize = blocksize;
  47. }
  48.  
  49. @Override
  50. public void run() {
  51.  
  52. BufferedInputStream bis = null;
  53. RandomAccessFile raf = null;
  54.  
  55. try {
  56. URLConnection conn = downloadUrl.openConnection();
  57. conn.setAllowUserInteraction(true);
  58.  
  59. int startPos = blockSize * (threadId - 1);//开始位置
  60. int endPos = blockSize * threadId - 1;//结束位置
  61. //设置当前线程下载的起点、终点
  62. conn.setRequestProperty("Range", "bytes=" + startPos + "-" + endPos);
  63. System.out.println(Thread.currentThread().getName() + " bytes="
  64. + startPos + "-" + endPos);
  65.  
  66. byte[] buffer = new byte[1024];
  67. bis = new BufferedInputStream(conn.getInputStream());
  68.  
  69. raf = new RandomAccessFile(file, "rwd");
  70. raf.seek(startPos);
  71. int len;
  72. while ((len = bis.read(buffer, 0, 1024)) != -1) {
  73. raf.write(buffer, 0, len);
  74. downloadLength += len;
  75. }
  76. isCompleted = true;
  77. Log.d(TAG, "current thread task has finished,all size:"
  78. + downloadLength);
  79.  
  80. } catch (IOException e) {
  81. e.printStackTrace();
  82. } finally {
  83. if (bis != null) {
  84. try {
  85. bis.close();
  86. } catch (IOException e) {
  87. e.printStackTrace();
  88. }
  89. }
  90. if (raf != null) {
  91. try {
  92. raf.close();
  93. } catch (IOException e) {
  94. e.printStackTrace();
  95. }
  96. }
  97. }
  98. }
  99.  
  100. /**
  101. * 线程文件是否下载完毕
  102. */
  103. public boolean isCompleted() {
  104. return isCompleted;
  105. }
  106.  
  107. /**
  108. * 线程下载文件长度
  109. */
  110. public int getDownloadLength() {
  111. return downloadLength;
  112. }
  113.  
  114. }

效果图:

Log控制台:

可以看到文件总大小、我们创建的5个线程每个负责下载的区间

SD卡:

断点续传

这个用到了数据存储保存当前每个线程下载文件的长度,等下一次再下载时读取,网上有成熟的案例,就不再造轮子了,资源里我打包了自己的项目和带断点续传的项目(别人的),大家可以下载下来直接用,全部免费:

http://download.csdn.net/detail/mad1989/7727133

Android实现网络多线程文件下载的更多相关文章

  1. Android 实现网络多线程APK文件下载

    (转自:http://blog.csdn.net/mad1989/article/details/38421465) 实现原理 (1)首先获得下载文件的长度,然后设置本地文件的长度. (2)根据文件长 ...

  2. Android实现网络多线程断点续传下载(转)

    本示例介绍在Android平台下通过HTTP协议实现断点续传下载. 我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多 ...

  3. Android实现网络多线程断点续传下载

    本示例介绍在Android平台下通过HTTP协议实现断点续传下载. 我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多 ...

  4. Android网络多线程断点续传下载

    本示例介绍在Android平台下通过HTTP协议实现断点续传下载. 我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多 ...

  5. Android多线程文件下载器

    本应用实现的是输入文件的网络的地址,点击button開始下载,下载过程中有进度条和后面的文本提示进度, 下载过程中button不可点击,防止反复的下载,完成下载后会进行Toast的提示显示, 而且回复 ...

  6. 基于Retrofit+RxJava的Android分层网络请求框架

    目前已经有不少Android客户端在使用Retrofit+RxJava实现网络请求了,相比于xUtils,Volley等网络访问框架,其具有网络访问效率高(基于OkHttp).内存占用少.代码量小以及 ...

  7. Android okHttp网络请求之文件上传下载

    前言: 前面介绍了基于okHttp的get.post基本使用(http://www.cnblogs.com/whoislcj/p/5526431.html),今天来实现一下基于okHttp的文件上传. ...

  8. Android实现网络访问

    Android实现网络访问 开发工具:Andorid Studio 1.3 运行环境:Android 4.4 KitKat 工程内容 1) 熟练使用HttpURLConnection访问WebServ ...

  9. Android Learning:多线程与异步消息处理机制

    在最近学习Android项目源码的过程中,遇到了很多多线程以及异步消息处理的机制.由于之前对这块的知识只是浅尝辄止,并没有系统的理解.但是工程中反复出现让我意识到这个知识的重要性.所以我整理出这篇博客 ...

随机推荐

  1. Python 文件读和写

  2. load d3dcompiler_46.dll failed

    https://gist.github.com/rygorous/7936047 编shader的时候遇到这个warning不知道是不是什么隐患..从今天开始要做新项目了 尝试从同事那里要了这dll ...

  3. 常用的Python字符串常量

    下面是一些常用的Python字符串常量string.digits:包含0-9的字符串string.letters:包含所有大小写字母的字符串 string.lowercase:所有小写字母string ...

  4. 更改DEVExpress的Column的DisplayFormat为自定义的方法。

    更改DEVExpress的Column的DisplayFormat为自定义的方法. public partial class Form1 : XtraForm { public Form1() { I ...

  5. 高达渐出现效果Shader

    原地址: http://liweizhaolili.blog.163.com/blog/static/1623074420140591864/ 最近在玩游戏<高达破坏者>,里面的高达出现的 ...

  6. MySQL中文全文索引插件 mysqlcft 1.0.0 安装使用文档[原创]

    [文章+程序 作者:张宴 本文版本:v1.0 最后修改:2008.07.01 转载请注明原文链接:http://blog.zyan.cc/post/356/] MySQL在高并发连接.数据库记录数较多 ...

  7. Acdream1217 Cracking' RSA(高斯消元)

    题意:给你m个数(m<=100),每个数的素因子仅来自于前t(t<=100)个素数,问这m个数的非空子集里,满足子集里的数的积为完全平方数的有多少个. 一开始就想进去里典型的dp世界观里, ...

  8. hdu 3886 Final Kichiku “Lanlanshu” 数位DP

    思路: dp[i][j][k]:满足在字符串的j位,前一位数字是k. 代码如下: #include<iostream> #include<cstdio> #include< ...

  9. 今天来做一个PHP电影小爬虫。

    今天来做一个PHP电影小爬虫.我们来利用simple_html_dom的采集数据实例,这是一个PHP的库,上手很容易.simple_html_dom 可以很好的帮助我们利用php解析html文档.通过 ...

  10. C#遍历打印机

    #region 获取本机默认打印机名称 ArrayList al1=new ArrayLIst(); private static PrintDocument fPrintDocument = new ...