转载请注明出处:http://blog.csdn.net/lmj623565791/article/details/26994463

有个朋友须要个多线程如今的样例,就帮忙实现了。在此分享下~

先说下原理,原理明确了,事实上非常easy:

a、对于网络上的一个资源,首先发送一个请求,从返回的Content-Length中回去须要下载文件的大小。然后依据文件大小创建一个文件。

  1. this.fileSize = conn.getContentLength();// 依据响应获取文件大小
  2. File dir = new File(dirStr);
  3. this.localFile = new File(dir, filename);
  4. RandomAccessFile raf = new RandomAccessFile(this.localFile, "rw");
  5. raf.setLength(fileSize);
  6. raf.close();

b、依据线程数和文件大小。为每一个线程分配下载的字节区间,然后每一个线程向server发送请求。获取这段字节区间的文件内容。

  1. conn.setRequestProperty("Range", "bytes=" + startPos + "-"
  2. + endPos);// 设置获取实体数据的范围

c、利用RandomAccessFile的seek方法,多线程同一时候往一个文件里写入字节。

  1. raf.seek(startPos);
  2. while ((len = is.read(buf)) != -1)
  3. {
  4. raf.write(buf, 0, len);
  5. }

分析完了原理就非常easy了。我封装了一个类,利用这个类的实例进行下载。所需參数:下载资源的URI, 本地文件路径,线程的数量。

  1. package com.zhy.mutilthread_download;
  2.  
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.io.InputStream;
  6. import java.io.RandomAccessFile;
  7. import java.net.HttpURLConnection;
  8. import java.net.URL;
  9.  
  10. public class MultipartThreadDownloador
  11. {
  12.  
  13. /**
  14. * 须要下载资源的地址
  15. */
  16. private String urlStr;
  17. /**
  18. * 下载的文件
  19. */
  20. private File localFile;
  21. /**
  22. * 须要下载文件的存放的本地目录路径
  23. */
  24. private String dirStr;
  25. /**
  26. * 存储到本地的文件名称
  27. */
  28. private String filename;
  29.  
  30. /**
  31. * 开启的线程数量
  32. */
  33. private int threadCount;
  34. /**
  35. * 下载文件的大小
  36. */
  37. private long fileSize;
  38.  
  39. public MultipartThreadDownloador(String urlStr, String dirStr,
  40. String filename, int threadCount)
  41. {
  42. this.urlStr = urlStr;
  43. this.dirStr = dirStr;
  44. this.filename = filename;
  45. this.threadCount = threadCount;
  46. }
  47.  
  48. public void download() throws IOException
  49. {
  50. createFileByUrl();
  51.  
  52. /**
  53. * 计算每一个线程须要下载的数据长度
  54. */
  55. long block = fileSize % threadCount == 0 ? fileSize / threadCount
  56. : fileSize / threadCount + 1;
  57.  
  58. for (int i = 0; i < threadCount; i++)
  59. {
  60. long start = i * block;
  61. long end = start + block >= fileSize ? fileSize : start + block - 1;
  62.  
  63. new DownloadThread(new URL(urlStr), localFile, start, end).start();
  64. }
  65.  
  66. }
  67.  
  68. /**
  69. * 依据资源的URL获取资源的大小,以及在本地创建文件
  70. */
  71. public void createFileByUrl() throws IOException
  72. {
  73. URL url = new URL(urlStr);
  74. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  75. conn.setConnectTimeout(15 * 1000);
  76. conn.setRequestMethod("GET");
  77. conn.setRequestProperty(
  78. "Accept",
  79. "image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*");
  80. conn.setRequestProperty("Accept-Language", "zh-CN");
  81. conn.setRequestProperty("Referer", urlStr);
  82. conn.setRequestProperty("Charset", "UTF-8");
  83. conn.setRequestProperty(
  84. "User-Agent",
  85. "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0; .NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
  86. conn.setRequestProperty("Connection", "Keep-Alive");
  87. conn.connect();
  88.  
  89. if (conn.getResponseCode() == 200)
  90. {
  91. this.fileSize = conn.getContentLength();// 依据响应获取文件大小
  92. if (fileSize <= 0)
  93. throw new RuntimeException(
  94. "the file that you download has a wrong size ... ");
  95. File dir = new File(dirStr);
  96. if (!dir.exists())
  97. dir.mkdirs();
  98. this.localFile = new File(dir, filename);
  99. RandomAccessFile raf = new RandomAccessFile(this.localFile, "rw");
  100. raf.setLength(fileSize);
  101. raf.close();
  102.  
  103. System.out.println("须要下载的文件大小为 :" + this.fileSize + " , 存储位置为: "
  104. + dirStr + "/" + filename);
  105.  
  106. } else
  107. {
  108. throw new RuntimeException("url that you conneted has error ...");
  109. }
  110. }
  111.  
  112. private class DownloadThread extends Thread
  113. {
  114. /**
  115. * 下载文件的URI
  116. */
  117. private URL url;
  118. /**
  119. * 存的本地路径
  120. */
  121. private File localFile;
  122. /**
  123. * 是否结束
  124. */
  125. private boolean isFinish;
  126. /**
  127. * 開始的位置
  128. */
  129. private Long startPos;
  130. /**
  131. * 结束位置
  132. */
  133. private Long endPos;
  134.  
  135. public DownloadThread(URL url, File savefile, Long startPos, Long endPos)
  136. {
  137. this.url = url;
  138. this.localFile = savefile;
  139. this.startPos = startPos;
  140. this.endPos = endPos;
  141. }
  142.  
  143. @Override
  144. public void run()
  145. {
  146. System.out.println(Thread.currentThread().getName() + "開始下载...");
  147. try
  148. {
  149. HttpURLConnection conn = (HttpURLConnection) url
  150. .openConnection();
  151. conn.setConnectTimeout(15 * 1000);
  152. conn.setRequestMethod("GET");
  153. conn.setRequestProperty(
  154. "Accept",
  155. "image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*");
  156. conn.setRequestProperty("Accept-Language", "zh-CN");
  157. conn.setRequestProperty("Referer", url.toString());
  158. conn.setRequestProperty("Charset", "UTF-8");
  159. conn.setRequestProperty("Range", "bytes=" + startPos + "-"
  160. + endPos);// 设置获取实体数据的范围
  161.  
  162. conn.setRequestProperty(
  163. "User-Agent",
  164. "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0; .NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
  165. conn.setRequestProperty("Connection", "Keep-Alive");
  166. conn.connect();
  167.  
  168. /**
  169. * 代表server已经成功处理了部分GET请求
  170. */
  171. if (conn.getResponseCode() == 206)
  172. {
  173. InputStream is = conn.getInputStream();
  174. int len = 0;
  175. byte[] buf = new byte[1024];
  176.  
  177. RandomAccessFile raf = new RandomAccessFile(localFile,
  178. "rwd");
  179. raf.seek(startPos);
  180. while ((len = is.read(buf)) != -1)
  181. {
  182. raf.write(buf, 0, len);
  183. }
  184. raf.close();
  185. is.close();
  186. System.out.println(Thread.currentThread().getName()
  187. + "完毕下载 : " + startPos + " -- " + endPos);
  188. this.isFinish = true;
  189. } else
  190. {
  191. throw new RuntimeException(
  192. "url that you conneted has error ...");
  193. }
  194. } catch (IOException e)
  195. {
  196. e.printStackTrace();
  197. }
  198. }
  199.  
  200. }
  201.  
  202. }

createFileByUrl方法,就是我们上述的原理的步骤1。得到文件大小和创建本地文件。

我在程序使用了一个内部类DownloadThread继承Thread,专门负责下载。download()方法,依据线程数量和文件大小计算每一个线程须要下载的字节区间。然后开启线程去下载。

server端:我就扔了几个文件在Tomcat根文件夹做实验,以下是測试代码:

  1. package com.zhy.mutilthread_download;
  2.  
  3. import java.io.IOException;
  4.  
  5. public class Test
  6. {
  7.  
  8. public static void main(String[] args)
  9. {
  10. try
  11. {
  12. new MultipartThreadDownloador("http://localhost:8080/nexus.zip",
  13. "f:/backup/nexus", "nexus.zip", 2).download();
  14. } catch (IOException e)
  15. {
  16. e.printStackTrace();
  17. }
  18.  
  19. }
  20. }

输出结果:

  1. 须要下载的文件大小为 :31143237 , 存储位置为: f:/backup/nexus/nexus.zip
  2. Thread-1開始下载...
  3. Thread-2開始下载...
  4. Thread-3開始下载...
  5. Thread-4開始下载...
  6. Thread-4完毕下载 23357430 -- 31143237
  7. Thread-2完毕下载 7785810 -- 15571619
  8. Thread-1完毕下载 0 -- 7785809
  9. Thread-3完毕下载 15571620 -- 23357429

截图:

ok,多线程下载介绍完成,假设代码设计不合理,以及方法使用错误。欢迎各位留言。。。

源代码点击下载

版权声明:本文博主原创文章,博客,未经同意不得转载。

Java / Android H基于ttp多线程下载的实现的更多相关文章

  1. Android学习记录(6)—将java中的多线程下载移植到Android中(即多线程下载在Android中的使用)③

    在这一节中,我们就来讲多线程下载以及断点续传在android中怎么使用,前两节是为本节做准备的,没有看前两节的同学,最好看完前面的两篇文章再来看这篇.其实在android端的应用和java基本上是差不 ...

  2. 160G 视频教程(Java+Android+项目视频)免费下载

    我不喜欢多说没用,直接给下载链接,进去直接下载,下载不动的联系网站客服解决!我只和我的好朋友们分享好的视频教程 http://edu.csdn.net/main/video.shtml 视频教程目录过 ...

  3. Android开发之多线程下载、断点续传、进度条和文本显示

    代码实现了在Android环境下的多线程下载.断点续传.进度条显示和文本显示百分数: import java.io.BufferedReader; import java.io.File; impor ...

  4. Android中多线程下载列表的封装实现(含进度反馈)

    来源:http://blog.csdn.net/u011638883/article/details/17347015 实现了一下Android中的文件多线程下载模块,支持自定义线程数.断点续传.下载 ...

  5. Java / Android 基于Http的多线程下载的实现

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/26994463 有个朋友需要个多线程现在的例子,就帮忙实现了,在此分享下~ 先说下 ...

  6. Android学习记录(4)—在java中学习多线程下载的基本原理和基本用法①

    多线程下载在我们生活中非常常见,比如迅雷就是我们常用的多线程的下载工具,当然还有断点续传,断点续传我们在下一节来讲,android手机端下载文件时也可以用多线程下载,我们这里是在java中写一个测试, ...

  7. Android学习记录(5)—在java中学习多线程下载之断点续传②

    在上一节中我们学习了在java中学习多线程下载的基本原理和基本用法,我们并没有讲多线程的断点续传,那么这一节我们就接着上一节来讲断点续传,断点续传的重要性不言而喻,可以不用重复下载,也可以节省时间,实 ...

  8. android程序---->android多线程下载(一)

    多线程下载是加快下载速度的一种方式,通过开启多个线程去执行一个任务,可以使任务的执行速度变快.多线程的任务下载时常都会使用得到断点续传下载,就是我们在一次下载未结束时退出下载,第二次下载时会接着第一次 ...

  9. android程序---->android多线程下载(二)

    上篇我们讲到了android中下载的断点续传问题,今天我们开始学习下载的多线程问题.本次的多线程源码下载:androdi中多线程下载的实现代码.有关断点续传的问题,请参见博客:android程序--- ...

随机推荐

  1. mini2440裸机试炼之——DMA直接存取 实现Uart(串口)通信

    这个仅仅能作为自己初步了解MDA的开门篇 实现功能: 将字符串数据通过DMA0通道传递给UTXH0,然后在终端 显示.传输数据完后.DMA0产生中断,beep声, LED亮. DMA基本知识 计算机系 ...

  2. AES算法简介

    AES算法简介 一. AES的结构 1.总体结构 明文分组的长度为128位即16字节,密钥长度可以为16,24或者32字节(128,192,256位).根据密钥的长度,算法被称为AES-128,AES ...

  3. 关于css3的rgba

    在rgba之前,我们应该知道rgb.它就是红色R+绿色G+蓝色B.那rgba是什么?简单的说就是在rgb的基础之上加上一个通道alpha.他的语法如下: r    红色值.正整数(0~255) | 百 ...

  4. PL/SQL破解方法(不需要注册码)

    打开注册表在run下输入regedit删除1.HKEY_CURRENT_USER/Software/Allround Automations2.HKEY_CURRENT_USER/Software/M ...

  5. sealed 修饰符

    当对一个类应用 sealed 修饰符时,此修饰符会阻止其他类从该类继承. 在下面的示例中,类 B 从类 A 继承,但是任何类都不能从类 B 继承. class A {} sealed class B ...

  6. Linux下Tomcat的安装配置 去掉应用名称

    http://blog.csdn.net/zhuying_linux/article/details/6583096/ Tomcat下为每个Web应用配置不同的访问端口 http://www.linu ...

  7. 关于Repeater中绑定的控件不触发ItemCommand事件

    今天遇到 在repeater 中使用一个button,点击button然后跳转另外一个页面. html. <asp:Repeater ID="repeater" runat= ...

  8. [0] Tornado Todo 开篇

    参考自: python: tornado例子 Github地址:tornado_todo 开发环境: Python包的安装: 首先安装 pip: sudo apt-get install python ...

  9. 利用Azure高级存储搭建高性能Linux服务器(1)

    目前Azure针对虚拟机提供两种类型的存储,一种是标准存储,基于HDD的,一种是高性能存储Premium Storage(在下文中简称PS),基于SSD的.针对用户高性能,低延迟,I/O密集型的应用, ...

  10. 解决Sqlite UTF-8中文数据格式在DOS窗口下中文乱码

    问题如下图所示: 右键查看DOS属性:点击选项->当前代码页为GBK 只要将当前代码页编码值设置为UTF-8就可以了: ‘退出sqlite:.exit 输入:chcp 65001回车 右键属性: ...