目录

1       大概思路... 1

2       Nginx集群之WCF大文件上传及下载... 1

3       BasicHttpBinding相关配置解析... 2

4       编写WCF服务、客户端程序... 3

5       URL保留项... 8

6       部署WCF服务程序到局域网内1台PC机... 8

7       Nginx集群配置搭建... 9

8       WCF客户端程序的运行结果... 11

9       总结... 13

1       大概思路

l  Nginx集群之WCF大文件上传及下载

l  BasicHttpBinding相关配置解析

transferMode、messageEncoding、maxReceivedMessageSize、receiveTimeout、sendTimeout

l  编写WCF服务、客户端程序

l  URL保留项

l  部署WCF服务程序到局域网内1台PC机

l  Nginx集群配置搭建

l  WCF客户端程序的运行结果

l  总结

2       Nginx集群之WCF大文件上传及下载

Nginx的匹配规则,很容易帮助我们划分WCF服务的网段,从而实现企业数据信息系统多区域划分,如小数据的微服务、传输数据文件的服务、即时通信服务、或者邮件服务,相当于构建了一条企业内部信息化的数据总线(DataBus)。

以下是本文讲述的主要结构图:

客户端以BasicHttpBinding访问Nginx的域名zhyongfeng.com/fileupload,然后Nginx进行负载均衡,将消息分发到后端任意一台WCF上传下载服务器的PC机,然后进行上传文件至“冷数据”处,又从“冷数据”处下载文件。

针对“冷数据”可以划分目录,建立单独的FTP服务器及WCF服务器,进行操作处理。如下图所示:

以下是WCF上传下载服务器的架构:

3       BasicHttpBinding相关配置解析

basicHttpBinding相关配置,具体参考:

https://msdn.microsoft.com/zh-cn/library/system.servicemodel.basichttpbinding.aspx

以下是主要应用到的配置

transferMode

获取或设置一个值,该值指示是通过缓冲处理还是流处理来发送消息。(继承自 HttpBindingBase。)

messageEncoding

获取或设置是使用 MTOM 还是文本对 SOAP 消息进行编码。

maxReceivedMessageSize

获取或设置最大大小,以字节为单位,可以使用此绑定配置的通道上接收的消息。(继承自 HttpBindingBase。)

receiveTimeout

获取或设置连接在撤消之前保持非活动状态的最大时间间隔,在此时间间隔内未接收任何应用程序消息。(继承自 Binding。)

sendTimeout

获取或设置在传输引发异常之前可用于完成写入操作的时间间隔。(继承自 Binding。)

针对MTOM编码和异步调用的性能改善,可以参照论文:

利用MTOM编码和异步调用改进流传输的性能.pdf

4       编写WCF服务、客户端程序

l  WCF服务程序

Program.cs

using System.ServiceModel;
using Service;
using System; namespace FileTransferHosting
{
class Program
{
static void Main(string[] args)
{
using (ServiceHost host = new ServiceHost(typeof(FileTransferOperation)))
{
host.Opened += delegate
{
Console.WriteLine(host.Description.Endpoints[].Address.Uri + "已经启动,按任意键终止服务!");
}; host.Open();
Console.Read();
}
}
}
}

FileTransferOperation.cs

using Service.Interface;
using System;
using System.IO;
using System.ServiceModel; namespace Service
{
[ServiceBehavior(ConcurrencyMode = ConcurrencyMode.Multiple)]
public class FileTransferOperation : IFileTransferOperation
{
/// <summary>
/// 上传文件
/// </summary>
/// <param name="remoteFile"></param>
public void UploadLoad(RemoteFileInfo remoteFile)
{
StreamToFile(remoteFile);
} /// <summary>
/// 写文件
/// </summary>
/// <param name="remoteFile"></param>
public void StreamToFile(RemoteFileInfo remoteFile)
{
string fileFullPath = Path.Combine(System.Environment.CurrentDirectory, remoteFile.FileName);
Stream sourceStream = remoteFile.FileByteStream;
if (sourceStream == null) { return; }
if (!sourceStream.CanRead) { return; }
//创建文件流,读取流中的数据生成文件
using (FileStream fs = new FileStream(fileFullPath, FileMode.Create, FileAccess.Write, FileShare.None))
{
const int bufferLength = ;
byte[] myBuffer = new byte[bufferLength];//数据缓冲区
int count;
while ((count = sourceStream.Read(myBuffer, , bufferLength)) > )
{
fs.Write(myBuffer, , count);
}
fs.Close();
sourceStream.Close();
}
} /// <summary>
/// 下载文件
/// </summary>
/// <param name="fileName"></param>
/// <returns></returns>
public Stream DownLoad(string fileName)
{
string fileFullPath = Path.Combine(System.Environment.CurrentDirectory, fileName);
if (!File.Exists(fileFullPath))//判断文件是否存在
{
return null;
}
try
{
Stream myStream = File.OpenRead(fileFullPath);
return myStream;
}
catch { return null; }
}
}
}

服务端配置文件:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.serviceModel>
<behaviors>
<serviceBehaviors>
<behavior name="FileTransferBehavior">
<serviceMetadata httpGetEnabled="true" />
<serviceDebug includeExceptionDetailInFaults="true" />
</behavior>
</serviceBehaviors>
</behaviors>
<bindings>
<basicHttpBinding>
<binding name="FileTransferdBinding" receiveTimeout="00:30:00" sendTimeout="00:30:00" maxReceivedMessageSize="" transferMode="Streamed"
messageEncoding="Mtom" />
</basicHttpBinding>
</bindings>
<services>
<service behaviorConfiguration="FileTransferBehavior" name="Service.FileTransferOperation">
<endpoint binding="basicHttpBinding" bindingConfiguration="FileTransferdBinding"
contract="Service.Interface.IFileTransferOperation" />
<host>
<baseAddresses>
<add baseAddress="http://127.0.0.1:5600/fileupload" />
</baseAddresses>
</host>
</service>
</services>
</system.serviceModel>
</configuration>

l  客户端程序

Program.cs

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using FileTransferClient.FileTransferService; namespace FileTransferClient
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("请输入文件完整路径:");
string fullFilePath = Console.ReadLine().Trim(); using (FileTransferOperationClient proxy=new FileTransferOperationClient())
{
DateTime datetime1 = DateTime.Now;
byte[] buffer = System.IO.File.ReadAllBytes(fullFilePath);
Stream streamBuffer= new MemoryStream(buffer);
//上传文件
proxy.UploadLoad(buffer.Length, System.IO.Path.GetFileName(fullFilePath), streamBuffer);
OutPutDiffTime(DateTime.Now, datetime1, "上传成功"); Console.WriteLine("开始下载文件:");
DateTime datetime2 = DateTime.Now;
string filename = System.IO.Path.GetFileName(fullFilePath);
//下载文件
Stream sourceStream = proxy.DownLoad(filename);
if (sourceStream == null) { return; }
if (!sourceStream.CanRead) { return; }
//创建文件流,读取流中的数据生成文件
using (FileStream fs = new FileStream(Path.Combine(System.Environment.CurrentDirectory, filename), FileMode.Create, FileAccess.Write, FileShare.None))
{
const int bufferLength = ;
//数据缓冲区
byte[] myBuffer = new byte[bufferLength];
int count;
while ((count = sourceStream.Read(myBuffer, , bufferLength)) > )
{
fs.Write(myBuffer, , count);
}
fs.Close();
sourceStream.Close();
}
OutPutDiffTime(DateTime.Now, datetime2, "下载成功");
Console.Read();
}
}
public static void OutPutDiffTime(DateTime datetime2,DateTime datetime1,string mesg)
{
TimeSpan ts = (datetime2 - datetime1);
string dateDiff = ts.Hours.ToString() + "小时"
+ ts.Minutes.ToString() + "分钟"
+ ts.Seconds.ToString() + "秒";
Console.WriteLine(String.Format("花费了{0},{1}", dateDiff, mesg));
}
}
}

客户端配置文件(注意这里客户端transferMode=”Streamed”要采用流式数据进行处理,并且将endpoint address改为” http://zhyongfeng.com/fileupload”):

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<system.serviceModel>
<bindings>
<basicHttpBinding>
<binding name="BasicHttpBinding_IFileTransferOperation" receiveTimeout="00:30:00"
sendTimeout="00:30:00" maxReceivedMessageSize="" transferMode="Streamed"
messageEncoding="Mtom" />
</basicHttpBinding>
</bindings>
<client>
<endpoint address="http://zhyongfeng.com/fileupload" binding="basicHttpBinding"
bindingConfiguration="BasicHttpBinding_IFileTransferOperation"
contract="FileTransferService.IFileTransferOperation" name="BasicHttpBinding_IFileTransferOperation" />
</client>
</system.serviceModel>
</configuration>

如下图所示:

5       URL保留项

详见:http://www.cnblogs.com/yongfeng/p/7851039.html

6       部署WCF服务程序到局域网内1台PC机

远程部署WCF服务端程序到PC机

7       Nginx集群配置搭建

通过自主义域名zhyongfeng.com:80端口进行负载均衡集群访问,则访问C:\Windows\System32\drivers\etc\hosts,添加下列“本机IP 自定义的域名”:

10.93.85.66    zhyongfeng.com

使用Nginx匹配原则针对WCF部署的1台PC机配置如下:

worker_processes  1;
events {
worker_connections 1024;
}
http {
include mime.types;
default_type application/octet-stream;
sendfile on;
keepalive_timeout 65; upstream zhyongfeng.com {
server 10.92.202.56:5600;
server 10.92.202.57:5700;
server 10.92.202.58:5800;
}
server {
listen 80;
server_name zhyongfeng.com;
location / {
proxy_pass http://zhyongfeng.com;
proxy_connect_timeout 10s;
}
location /fileupload {
proxy_pass http://10.92.202.56:5600/fileupload;
proxy_connect_timeout 10s;
}
}
}

运行CMD:

D:\DTLDownLoads\nginx-1.10.2>start nginx

D:\DTLDownLoads\nginx-1.10.2>nginx -s reload

访问WCF服务端:http://zhyongfeng.com/fileupload,运行结果:

8        WCF客户端程序的运行结果

启动WCF客户端程序,直接上传Linux的ubuntu系统iso镜像,总大小为1.5G左右,上传时间局域网(服务器线路是10Gpbs)花费7分钟左右,下载时间大概是3分钟。

客户端服务器的网络状况,运行效果如下图:

9       总结

Nginx的匹配原则能够有效的分配URL,将流式数据分发给相应的服务处理,并且在局域网内能够支持较大的上传下载功能。通过BasicHttpBinding的相关配置,能够控制流式数据上传大小,同时支持流式数据的下载功能,达到WCF大文件上传及下载的效果。

当然,具体的应用场景,还是要结合数据大小而言论的,这里只是提供一个解决方案的参考。

例如一些处理视频文件每天都是1TB,处理PB级的大数据文件,还是建议使用hadoop的HDFS实现比较好。

另一方面,数据大小达若干MB或几KB的发票、报表文件,这些文件主要在多而不在大,hadoop的MapReduce显然有点大材小用了,可以采用“业务领域-年-月-日”的方式,建立自己一套数据结构存储方式。

源代码下载:

http://download.csdn.net/download/ruby_matlab/10131307

PDF下载:

Nginx集群之WCF大文件上传及下载(支持6G传输).pdf

Nginx集群之WCF大文件上传及下载(支持6G传输)的更多相关文章

  1. 全网最简单的大文件上传与下载代码实现(React+Go)

    前言 前段时间我需要实现大文件上传的需求,在网上查找了很多资料,并且也发现已经有很多优秀的博客讲了大文件上传下载这个功能. 我的项目是个比较简单的项目,并没有采用特别复杂的实现方式,所以我这篇文章的目 ...

  2. PHP实现大文件上传和下载

    一提到大文件上传,首先想到的是啥??? 没错,就是修改php.ini文件里的上传限制,那就是upload_max_filesize.修改成合适参数我们就可以进行愉快的上传文件了.当然啦,这是一般情况下 ...

  3. java实现大文件上传和下载

    [文件上传和下载]是很多系统必备功能, 比如PM\OA\ERP等:系统中常见的开发模式有B/S和C/S,而前者主要是通过浏览器来访问web服务器,一般采用七层协议中的[应用层http]进行数据传输,后 ...

  4. ASP.NET实现大文件上传和下载

    总结一下大文件分片上传和断点续传的问题.因为文件过大(比如1G以上),必须要考虑上传过程网络中断的情况.http的网络请求中本身就已经具备了分片上传功能,当传输的文件比较大时,http协议自动会将文件 ...

  5. JSP实现大文件上传和下载

    javaweb上传文件 上传文件的jsp中的部分 上传文件同样可以使用form表单向后端发请求,也可以使用 ajax向后端发请求 1.通过form表单向后端发送请求 <form id=" ...

  6. WEB实现大文件上传和下载

    我们平时经常做的是上传文件,上传文件夹与上传文件类似,但也有一些不同之处,这次做了上传文件夹就记录下以备后用. 这次项目的需求: 支持大文件的上传和续传,要求续传支持所有浏览器,包括ie6,ie7,i ...

  7. Nginx集群之WCF分布式局域网应用

    目录 1       大概思路... 1 2       Nginx集群WCF分布式局域网结构图... 1 3       关于WCF的BasicHttpBinding. 1 4       编写WC ...

  8. Nginx集群之WCF分布式身份验证(支持Soap)

    目录 1       大概思路... 1 2       Nginx集群之WCF分布式身份验证... 1 3       BasicHttpBinding.ws2007HttpBinding. 2 4 ...

  9. Nginx集群之WCF分布式消息队列

    目录 1       大概思路... 1 2       Nginx集群之WCF分布式消息队列... 1 3       MSMQ消息队列... 2 4       编写WCF服务.客户端程序... ...

随机推荐

  1. 【GISer&&Painter】GISer

    基于上一篇OpenGL的渲染原理,这两周又陆续接触了一些关于WebGL绘图的一些内容,因为刚入门,很多东西又很晦涩,所以特意花了小半天的时间整理了一下,特此记录. 一   画布和画笔:创建Canvas ...

  2. Solr6.1.0Windows安装步骤

    一. 环境 solr 6.1.0  下载地址 http://archive.apache.org/dist/lucene/solr/6.1.0/ jdk 1.8 tomcat8 二. 安装solr到t ...

  3. Java VS .NET:Java与.NET的特点对比

    一.前言 为什么要写Java跟.NET对比? .NET出生之后就带着Java的影子.从模仿到创新,.NET平台也越来越成熟.他们不同的支持者也经常因为孰弱孰强的问题争论不休.但是本文并不是为了一分高下 ...

  4. How to support comparators in our sort implementations?

    上图是普林斯顿算法课part1.Mergesort章节给出的参考代码,可以发现这个代码有三处警告.造成的隐患就是我们无法在类型检查时发现送入sort()函数的数组元素类型和Comparator的泛型不 ...

  5. 将STM32 iap hex文件与app hex文件合并为一个hex文件

    日前公司产品需要增加远程升级功能,boot loader程序写好后交予生产部门使用时他们反馈每个产品程序需要刷写两次(一个boot loader 一个app程序),生产进度变慢浪费时间,于是乎研究如何 ...

  6. Ubuntu下关闭防火墙

    默认情况下ubuntu无firewall,除非你自己安装了,怎么装的就怎么删呗. . 假设是已启用的自备的iptables 删了即可了 sudo apt-get remove iptables.

  7. 如何导入外部的源码到eclipse中

    用struts,spring等框架开发也有两年的时间了,一直很少去阅读其源码,每次在eclipse编码的过程中想要看某一个类的源码,ctrl点击总是出现source not found的提示,也没有去 ...

  8. gunicorn syncworker 源码解析

    gunicorn支持不同的worker类型,同步或者异步,异步的话包括基于gevent.基于eventlet.基于Aiohttp(python版本需要大于3.3),也有多线程的版本.下面是gunico ...

  9. NanUI文档 - 使用网页来设计整个窗口

    NanUI文档目录 NanUI简介 开始使用NanUI 打包并使用内嵌式的HTML/CSS/JS资源 使用网页来设计整个窗口 如何实现C#与Javascript相互掉用(待更新...) 如何处理Nan ...

  10. 「mysql优化专题」视图应用竟然还可以这么优化?不得不收藏(8)

    一.视图概述(技术文): (1)什么是视图? 视图是基于 SQL 语句的结果集的可视化的表. 视图包含行和列,就像一个真实的表.视图中的字段就是来自一个或多个数据库中的真实的表中的字段.视图并不在数据 ...