多线程下载文件

作者:追风剑情 发布于:2016-6-30 10:33 分类:C#

示例代码

  1. using System;
  2. using System.IO;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading;
  7. using System.Net;
  8.  
  9. namespace ThreadPoolTest
  10. {
  11. class Program
  12. {
  13. static void Main(string[] args)
  14. {
  15. //此值默认为2(即,同时只能有2条http并发请求),这句必须在使用Http请求前加上,才能修改并发请求数。
  16. ServicePointManager.DefaultConnectionLimit = 200;
  17. //显示CPU内核数
  18. Console.WriteLine("Environment.ProcessorCount={0}", Environment.ProcessorCount);
  19.  
  20. string url = "http://192.168.1.211/trunk_pc_res/sgzs_voice.apk";
  21. int threadCount = Environment.ProcessorCount * 2; //线程数
  22. long fileSize = 0; //文件大小(单位: byte)
  23. //获取文件大小
  24. HttpWebRequest web_req = (HttpWebRequest)WebRequest.Create(url);
  25. HttpWebResponse web_resp = (HttpWebResponse)web_req.GetResponse();
  26. fileSize = web_resp.ContentLength;
  27. web_resp.Close();
  28. web_req.Abort();
  29.  
  30. //计算文件分块
  31. int blockSize = (int)(fileSize / threadCount); //平均分配
  32. int remainSize = (int)(fileSize % threadCount); //获取剩余的
  33.  
  34. Console.WriteLine("文件总长度: {0}, 分块长度: {1}, 剩余长度: {2}, 下载线程数: {3}", fileSize, blockSize, remainSize, threadCount);
  35.  
  36. ManualResetEvent manualResetEvent = new ManualResetEvent(false);
  37. try
  38. {
  39. HttpBlockDownFile.manualResetEvent = manualResetEvent;
  40. HttpBlockDownFile.threadMaxCount = threadCount;
  41. HttpBlockDownFile.threadCompleteCount = 0;
  42. HttpBlockDownFile.isCompleted = false;
  43. HttpBlockDownFile.downLoadSize = 0;
  44. HttpBlockDownFile.startTime = DateTime.Now;
  45. HttpBlockDownFile.mergeFileName = @"D:\\sgzs.apk";
  46. ThreadPool.SetMaxThreads(100, 100);
  47. ThreadPool.SetMinThreads(10, 10);
  48. int from, to=0;
  49. for (int i = 0; i < threadCount; i++)
  50. {
  51. from = i * blockSize;
  52. to = from + blockSize - 1;
  53. if (i == threadCount - 1)
  54. to += remainSize;
  55.  
  56. HttpBlockDownFile http = new HttpBlockDownFile();
  57. http.index = i;
  58. http.url = url;
  59. http.from = from;
  60. http.to = to;
  61. bool b = ThreadPool.QueueUserWorkItem(new WaitCallback(http.Run));
  62. Console.WriteLine("启动线程 ID={0}, success={1} 负责下载[{2}, {3}]", i, b, from, to);
  63. }
  64. }
  65. catch (NotSupportedException)
  66. {
  67. Console.WriteLine("These API's may fail when called on a non-Wind ows 2000 system.");
  68. }
  69.  
  70. Console.Read();
  71. }
  72. }
  73.  
  74. class HttpBlockDownFile
  75. {
  76. //最大线程数
  77. public static int threadMaxCount = 0;
  78. //完成线程数
  79. public static int threadCompleteCount = 0;
  80. //合并文件名
  81. public static string mergeFileName;
  82. //临时文件名格式
  83. public static string tmpFileFormat = @"D:\\sgzs_{0}.tmp";
  84. //当前已下载字节数
  85. public static volatile int downLoadSize = 0;
  86. //是否下载完成
  87. public static bool isCompleted = false;
  88.  
  89. public static ManualResetEvent manualResetEvent;
  90. public static DateTime startTime;
  91.  
  92. public int index;
  93. //下载地址
  94. public string url;
  95. //开始位置
  96. public int from;
  97. //结束位置
  98. public int to;
  99.  
  100. public void Run(Object state)
  101. {
  102. try
  103. {
  104. DateTime start_time = DateTime.Now;
  105. FileStream f = new FileStream(string.Format(tmpFileFormat, index), FileMode.Create);
  106. HttpWebRequest web_req = (HttpWebRequest)WebRequest.Create(url);
  107. web_req.AddRange(from, to);
  108. web_req.KeepAlive = false;
  109. web_req.Timeout = 10000;
  110.  
  111. HttpWebResponse web_resp = (HttpWebResponse)web_req.GetResponse();
  112. long contentLength = web_resp.ContentLength;
  113. byte[] read_bytes = new byte[1024*10];
  114. int read_size = web_resp.GetResponseStream().Read(read_bytes, 0, read_bytes.Length);
  115. int down_size = 0;
  116. while(read_size > 0){
  117. //Console.WriteLine("index={0}, down_size={1}, downLoadSize={2}", index, down_size, downLoadSize);
  118. Thread.Sleep(1);
  119. f.Write(read_bytes, 0, read_size);
  120. down_size += read_size;
  121. downLoadSize += read_size;
  122. read_size = web_resp.GetResponseStream().Read(read_bytes, 0, read_bytes.Length);
  123. }
  124. web_resp.Close();
  125. web_req.Abort();
  126. f.Flush(true);
  127. f.Close();
  128.  
  129. TimeSpan span = DateTime.Now - start_time;
  130.  
  131. Console.WriteLine("文件块下载完成: total: {0}, down_size: {1}, from: {2}, to: {3}, 耗时{4}秒, index={5}", contentLength, down_size, from, to, span.TotalSeconds, index);
  132.  
  133. Interlocked.Increment(ref threadCompleteCount);
  134. Console.WriteLine("threadCompleteCount: " + threadCompleteCount);
  135.  
  136. if (threadMaxCount == threadCompleteCount)
  137. {
  138. Complete();
  139. manualResetEvent.Set();
  140. }
  141. }
  142. catch (Exception e)
  143. {
  144. Console.WriteLine(e.ToString());
  145. }
  146. }
  147.  
  148. private void Complete()
  149. {
  150. Console.WriteLine("Complete()");
  151. TimeSpan span = DateTime.Now - startTime;
  152. Console.WriteLine("文件下载完成! 耗时{0}秒", span.TotalSeconds);
  153.  
  154. Console.WriteLine("开始合并文件...");
  155.  
  156. Stream mergeFile = new FileStream(mergeFileName, FileMode.Create);
  157. BinaryWriter mergeWriter = new BinaryWriter(mergeFile);
  158. for (int i = 0; i < threadMaxCount; i++)
  159. {
  160. string tmpFileName = string.Format(tmpFileFormat, i);
  161. using (FileStream fs = new FileStream(tmpFileName, FileMode.Open))
  162. {
  163. BinaryReader tempReader = new BinaryReader(fs);
  164. mergeWriter.Write(tempReader.ReadBytes((int)fs.Length));
  165. tempReader.Close();
  166. }
  167. File.Delete(tmpFileName);
  168. }
  169. mergeWriter.Close();
  170. isCompleted = true;
  171.  
  172. Console.WriteLine("文件合并完成 " + mergeFileName);
  173. }
  174. }
  175. }

 

运行测试1111.png

 

标签: C#

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号