最近项目需要做一个类似于迅雷的文件下载功能,这类需求可能比较常见,希望可以帮助到有需要的同学

要求:

1.支持断点传输

2. 多文件同时下载

3. 由于是客户内部试用,服务器只支持HTTP文件下载不支持FTP文件(并没有用户权限的要求)

由于根据服务器传来的Json字符串解析后进行下载,首先构建Json对象,推荐一个库Newtonsoft.Json

 public class DownloadFile
{
[JsonProperty("police_num")]
public string PoliceNumber { get; set; } [JsonProperty("equipment_num")]
public string EquipmentNumber { get; set; } [JsonProperty("upload_date")]
public DateTime UploadDateTime { get; set; } [JsonProperty("type")]
public string FileType { get; set; } [JsonProperty("file_name")]
public string FileName { get; set; } [JsonProperty("file_path")]
public string FilePath { get; set; }
}
JsonProperty不区分大小写
public class DownloadTask : ObservableObject
{
/// <summary>
/// </summary>
public static readonly int BufferSize = ; /// <summary>
/// 一次请求大小,每次请求100kb
/// </summary>
public static long Step = *; private int _process; /// <summary>
/// 下载文件参数
/// </summary>
public DownloadFile DownloadFile { get; set; } /// <summary>
/// 下载文件状态
/// </summary>
public SegmentState SegmentState { get; set; } /// <summary>
/// 默认存储路径
/// </summary>
public string DefaultDirectory { get; set; } /// <summary>
/// 当前进度
/// </summary>
public long CurrentSize { get; set; } /// <summary>
/// 文件流对象,用于生成文件
/// </summary>
public FileStream FileStream { get; set; } /// <summary>
/// 下载起始位置
/// </summary>
public long RangeFrom { get; set; } /// <summary>
/// 总大小
/// </summary>
public long TotalSize { get; set; } /// <summary>
/// 链接地址
/// </summary>
public string Url { get; set; } /// <summary>
/// 界面显示进度
/// </summary>
public int Process
{
get { return _process; }
set
{
_process = value;
RaisePropertyChanged("Process");
}
}
}
下载服务类,DownloaderService.cs,功能包括启动下载,暂停,针对文件是否存在进行初始化,获取文件大小方便界面显示
 public class DownloaderService
{ public void OnStart(DownloadTask downloadTask)
{
if (!InitService(downloadTask)) return;
var downloadThread = new Thread(BeginDownloadFile) { IsBackground = true };
downloadThread.Start(downloadTask);
} public void OnStop(DownloadTask downloadTask)
{
downloadTask.SegmentState = SegmentState.Paused;
} private bool InitService(DownloadTask downloadTask)
{
try
{
if (string.IsNullOrEmpty(downloadTask.Url))
{
return false;
} downloadTask.DownloadFile.FileName = downloadTask.Url.Substring(downloadTask.Url.LastIndexOf('/') + ); var fullName = Path.Combine(downloadTask.DefaultDirectory, downloadTask.DownloadFile.FileName); if (File.Exists(fullName))
{
if (downloadTask.TotalSize == )
{
GetTotalSize(downloadTask);
}
if (downloadTask.CurrentSize == downloadTask.TotalSize)
{
if (MessageBox.Show("文件已经存在是否覆盖此文件", "TCL", MessageBoxButton.YesNo, MessageBoxImage.Information) !=
MessageBoxResult.Yes)
return false;
File.Delete(fullName);
downloadTask.CurrentSize = ;
}
downloadTask.FileStream = new FileStream(fullName, FileMode.OpenOrCreate, FileAccess.ReadWrite);
downloadTask.SegmentState = SegmentState.Downloading;
return true;
}
downloadTask.SegmentState = SegmentState.Downloading;
downloadTask.FileStream = new FileStream(fullName, FileMode.OpenOrCreate, FileAccess.ReadWrite);
return true;
}
catch (Exception ex)
{
LogLightConsole.WriteLog(ex.ToString(), EunmMsgType.Error);
return false;
}
} private void BeginDownloadFile(object obj)
{
try
{
var downloadTask = (DownloadTask)obj;
while (true)
{
if (downloadTask.SegmentState == SegmentState.Paused)
{
downloadTask.FileStream.Close();
break;
}
//从0计数,需要减一
var from = downloadTask.CurrentSize;
if (from < )
{
from = ;
} var to = downloadTask.CurrentSize + DownloadTask.Step - ;
if (to >= downloadTask.TotalSize && downloadTask.TotalSize > )
{
to = downloadTask.TotalSize - ;
}
if (downloadTask.TotalSize == )
{
GetTotalSize(downloadTask);
}
if (from >= downloadTask.TotalSize || downloadTask.CurrentSize >= downloadTask.TotalSize)
{
downloadTask.SegmentState = SegmentState.Finished;
downloadTask.FileStream.Close();
downloadTask.Process = ;
return;
} var request = (HttpWebRequest)WebRequest.Create(downloadTask.Url);
//request.Method = "GET";
request.AddRange("bytes", from, to);
request.Timeout = * ;
var response = (HttpWebResponse)request.GetResponse();
var buffer = new byte[];
using (var stream = response.GetResponseStream())
{
if (stream != null)
{
var size = stream.Read(buffer, , buffer.Length);
while (size > )
{
//只将读出的字节写入文件
downloadTask.FileStream.Write(buffer, , size);
downloadTask.CurrentSize += size;
size = stream.Read(buffer, , buffer.Length);
downloadTask.Process = (int)(downloadTask.CurrentSize * / downloadTask.TotalSize);
downloadTask.FileStream.Flush();
}
}
//如果返回的response头中Content-Range值为空,说明服务器不支持Range属性,不支持断点续传,返回的是所有数据
if (response.Headers["Content-Range"] == null)
{
downloadTask.SegmentState = SegmentState.Finished;
}
}
} }
catch (Exception ex)
{
LogLightConsole.WriteLog("BeginDownloadFile" + ex, EunmMsgType.Error);
}
} /// <summary>
/// 获取文件总大小
/// </summary>
public void GetTotalSize(DownloadTask downloadTask)
{
var request = (HttpWebRequest)WebRequest.Create(downloadTask.Url);
//request.Method = "POST";
//request.Headers.Add("charset:" + "utf-8");
request.Timeout = * ;
var response = (HttpWebResponse)request.GetResponse();
downloadTask.TotalSize = response.ContentLength;
}
}

HTTP 1.1版本本身已经扩展了断点传输功能,但是这需要服务器同时也支持,这只是提供了一个简单的例子,具体的针对数据上下文以及成功失败的callback函数,就是做一下委托封装用就好,如果具体使用有什么问题可以私信我,谢谢大家支持!

response.Headers["Content-Range"] == null 说明服务器不支持断点传输

 

基于HTTP的断点传输的更多相关文章

  1. 基于HTTP在互联网传输敏感数据的消息摘要、签名与加密方案

    基于HTTP在互联网传输敏感数据的消息摘要.签名与加密方案 博客分类: 信息安全 Java 签名加密AESMD5HTTPS  一.关键词 HTTP,HTTPS,AES,SHA-1,MD5,消息摘要,数 ...

  2. curl基于URL的文件传输工具

    简介 cURL是一款开源的基于URL的文件传输工具,支持HTTP.HTTPS.FTP等协议,支持POST.cookie.认证.扩展头部.限速等特性. curl命令用途广泛,比如下载.发送http请求. ...

  3. C# Socket服务端与客户端通信(包含大文件的断点传输)

    步骤: 一.服务端的建立 1.服务端的项目建立以及页面布局 2.各功能按键的事件代码 1)传输类型说明以及全局变量 2)Socket通信服务端具体步骤:   (1)建立一个Socket   (2)接收 ...

  4. 基于epoll的TP传输层实现

    1. 抽象TP传输层设计 在使用epoll实现实际的传输层之前,先设计一个抽象的传输层,这个抽象的传输层是传输层实现的接口层. 接口层中一共有以下几个通用的类或者接口: (1)Socket:通用的套接 ...

  5. 基于SSL的WCF传输安全

    [实践]WCF传输安全1:前期准备之证书制作   [实践]WCF传输安全2:基于SSL的WCF匿名客户端   [实践]WCF传输安全3:基于SSL的WCF对客户端验证   [实践]WCF传输安全4:基 ...

  6. 基于网站地址URL传输session信息

    在php的学习中,会话是我们常常用到的,那今天我们就来详细讲讲会话中的session: 一.session的工作机制:当开启session后,服务器会在服务器中保存session文件,然后再浏览器保存 ...

  7. 基于TCP的字符串传输程序

    ---恢复内容开始--- LINUX中的网络编程是通过SOCKET接口来进行的. Socket(套接字) Socket相当于进行网络通信两端的插座,只要对方的Socket和自己的Socket有通信联接 ...

  8. 基于选择重传ARQ传输协议的数据重传机制方案设计

    原文链接: http://blog.csdn.net/pinghegood/article/details/7841281  1.背景 最近在项目中,由于使用TD网络传输数据,数据掉包严重,软件组老大 ...

  9. 基于socket的udp传输,socketserver模块,进程

    基于UDP的套接字 udp是无连接的,先启动哪一端都不会报错 socket.SOCK_DGRAM 数据报协议 udp不会发送空数据,什么都不输入直接发送也会有报头发过去 服务端 import sock ...

随机推荐

  1. HTML BOM Browser对象

    BOM:Browser Object Model,即浏览器对象模型,提供了独立于内容的.可以与浏览器窗口进行互动的对象结构. Browser对象:指BOM提供的多个对象,包括:Window.Navig ...

  2. Ngrok让你的本地Web应用暴露在公网上

    1.Ngrok介绍 Ngrok是一个反向代理,通过在公共的端点和本地运行的Web服务器之间建立一个安全的通道.Ngrok可捕获和分析所有通道上的流量,便于后期分析和重放.简单来说,利用 Ngrok可以 ...

  3. junit4进行单元测试

    一.前言 提供服务的时候,为了保证服务的正确性,有时候需要编写测试类验证其正确性和可用性.以前的做法都是自己简单写一个控制层,然后在控制层里调用服务并测试,这样做虽然能够达到测试的目的,但是太不专业了 ...

  4. 【知识必备】ezSQL,最好用的数据库操作类,让php操作sql更简单~

    最近用php做了点小东东,用上了ezSQL,感觉真的很ez,所以拿来跟大家分享一下~ ezSQL是一个非常好用的PHP数据库操作类.著名的开源博客WordPress的数据库操作就使用了ezSQL的My ...

  5. Android实现TCP断点上传,后台C#服务实现接收

    终端实现大文件上传一直都是比较难的技术,其中涉及到后端与前端的交互,稳定性和流量大小,而且实现原理每个人都有自己的想法,后端主流用的比较多的是Http来实现,因为大多实现过断点下载.但稳定性不能保证, ...

  6. 基于window7+caffe实现图像艺术风格转换style-transfer

    这个是在去年微博里面非常流行的,在git_hub上的代码是https://github.com/fzliu/style-transfer 比如这是梵高的画 这是你自己的照片 然后你想生成这样 怎么实现 ...

  7. 我理解的MVC

    前言 前一阶段对MVC模式及其衍生模式做了一番比较深入的研究和实践,这篇文章也算是一个阶段性的回顾和总结. 经典MVC模式 经典MVC模式中,M是指业务模型,V是指用户界面,C则是控制器,使用MVC的 ...

  8. 去IOE的一点反对意见以及其他

    某天在机场听见两老板在聊天,说到他们目前销售的报表老跟不上的问题,说要请一个人,专门合并和分析一些发过来的excel表格,我真想冲上去说,老板,你需要的是一个信息处理的系统,你需要咨询么.回来一直耿耿 ...

  9. 自定义控件之 圆形 / 圆角 ImageView

    一.问题在哪里? 问题来源于app开发中一个很常见的场景——用户头像要展示成圆的:       二.怎么搞? 机智的我,第一想法就是,切一张中间圆形透明.四周与底色相同.尺寸与头像相同的蒙板图片,盖在 ...

  10. Ajax.BeginForm方法 参数

    感谢博主 http://www.cnblogs.com/zzgblog/p/5454019.html toyoung 在Asp.Net的MVC中的语法,在Razor页面中使用,替代JQuery的Aja ...