C#用HTTP协议实现多线程文件下载和断点续传

(1) 多线程文件下载的总体思路是:把一个下载任务分配给多个线程,首先获取文件的长度,平分给多个线程,每个线程负责指定的文件块,待全部线程完成后,再把各个线程下载的临时文件合并。

(2)利用HttpWebResponse的ContentLength属性获取下载文件的总长度,利用HttpWebRequest的AddRange方法指定下载的文件数据范围。例如request.AddRange(500, 1000)表示下载的范围为从500字节开始到1000字节处结束。

(3)编写代码实现断点续传时,只需要记住已经成功下载的字节数,当需要从断点处继续下载时,根据已经下载的内容,重新指定下载的范围即可。

(4)   可以在各个线程中设置一个volatile类型的布尔值isPause,由主线程控制修改,当isPause为true的时候,各个线程停止下载文件,保存已经下载的字节数,当isPause为false的时候,各个线程读取上次停止时保存的已下载字节数,作为此次下载开始字节位置,继续下载文件。


下载子线程核心代码:

//接收线程
 public void Receive()
 {
    stopWatch.Reset();
    stopWatch.Start();
    AddStatus("线程" + threadIndex + "开始接收");
    while (true)
    {
      if (!isPause)
      {
        using (FileStream fs = new FileStream(TargetFileName, FileMode.OpenOrCreate))
        {
          try
          {
         HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create(SourceURI);
         request.AddRange(StartPosition, OriginalStartPosition + FileSize - 1);
         Stream stream = request.GetResponse().GetResponseStream();
         byte[] receiveBytes = new byte[512];
         int readBytes = stream.Read(receiveBytes, 0, receiveBytes.Length);
         while (!isPause && readBytes > 0)
           {
              fs.Write(receiveBytes, 0, readBytes);
              totalBytes += readBytes;
              readBytes = stream.Read(receiveBytes, 0, receiveBytes.Length);
              Thread.Sleep(100);
          }
        StartPosition = OriginalStartPosition + totalBytes;
        if (totalBytes < FileSize)
               AddStatus("线程" + threadIndex + "暂停接收!已接收字节数:" + Math.Ceiling(totalBytes / 1024.0f) + "KB");
       stream.Close();
       }
        catch (Exception ex)
       {
         AddStatus("线程" + threadIndex + "接收出错:" + ex.Message);
            }
          }
      }
      if (totalBytes >= FileSize)   //接收字节数等于文件块长度,代表已经接收完毕
              break;
       }
       ChangeStatus("线程" + threadIndex + "接收完毕", totalBytes);
       stopWatch.Stop();
       this.isFinished = true;
}

程序运行截图:



这次实验主要考察利用Http协议实现文件的多线程下载以及断点续传相关知识,实验中只实现了在一次运行过程中的“暂停-下载”功能,至于中途退出程序,再次打开程序能继续下载的断点续传还没来得及实现。我猜想实现的思路是每次打开程序是先去检测有没临时文件的存在,有的话就获取临时文件的大小,把各个临时文件的结束字节位置作为各个下载线程此次下载的起始位置,以此来实现断点续传。不知道这个思路对不对,有时间再去实践一下。




展开阅读全文

没有更多推荐了,返回首页