转:wcf大文件传输解决之道(1)
首先声明,文章思路源于MSDN中徐长龙老师的课程整理,加上自己的一些心得体会,先总结如下:
在应对与大文件传输的情况下,因为wcf默认采用的是缓存加载对象,也就是说将文件包一次性接受至缓存中,然后生成对象,显然对于大文件的传输,这种方式是不可取的,一般我们采用流传输或者更优秀的本本编码方式,在文本编码这一块我们一般采用w3c提出的MTOM传输机制,MTOM(Message Transmission Optimization Mechanism),是W3C的MTOM的消息传输优化机制,有效地发送的二进制数据和从Web服务方法。将消息传输优化机制 (MTOM) 消息编码与WSHttpBinding 一起使用。MTOM是一种机制,用来以原始字节形式传输包含SOAP消息的较大二进制附件,从而使所传输的消息较小。也就是说它对于文件的编码采取的文件头和文件包得传输方式,文件头定义文件格式,文件包采用SOAP原始的二进制信息,实现信息的优化,但这种方式在文件较小的时候相对于普通的文本编辑方式也是有耗损的,下面通过一个案例比较下:
新建方法,实现文本编码方式和MTOM方式编码的比较
///<summary>
/// 新建函数,用于测试moto方式对数据进行编码所生成的对象长度比较
///</summary>
///<param name="dataSize"></param>
static void CompareMessageSize(int dataSize)
{
byte[] binaryData = new byte[dataSize]; Message message = Message.CreateMessage(MessageVersion.Soap12WSAddressing10, "消息说明", binaryData);//创建一个soap消息 MessageBuffer buffer = message.CreateBufferedCopy(int.MaxValue);//新建一个消息缓存区域并将消息传入 int size = SizeofTextMessage(buffer.CreateMessage());
Console.WriteLine("text endcoding with a {0} byte playod:{1}", binaryData.Length, size); size = sizeofMotoMessage(buffer.CreateMessage());
Console.WriteLine("MTMO endcoding with a {0} byte playod:{1}", binaryData.Length, size);
Console.WriteLine(); message.Close();//关闭流
}
新建方法,实现MTOM方式编码
private static int sizeofMotoMessage(Message message)
{
//创建一个文本编码器
MessageEncodingBindingElement element = new MtomMessageEncodingBindingElement();
MessageEncoderFactory factory = element.CreateMessageEncoderFactory();
MessageEncoder encoder = factory.Encoder; MemoryStream sream = new MemoryStream(); //创建内存
encoder.WriteMessage(message, sream); //通过文本编辑器将消息写到内存中 int size = (int)sream.Length;
message.Close();
sream.Close();
return size;
}
新建另一个方法,实现默认文本方式编码
private static int SizeofTextMessage(Message message)
{
//创建一个Mtom编码器
MessageEncodingBindingElement element = new TextMessageEncodingBindingElement();
MessageEncoderFactory factory = element.CreateMessageEncoderFactory();
MessageEncoder encoder = factory.Encoder; MemoryStream sream = new MemoryStream(); //创建内存
encoder.WriteMessage(message, sream); //通过文本编辑器将消息写到内存中 int size = (int)sream.Length;
message.Close();
sream.Close();
return size; }
调用方式比较传输:
///比较文本编码方式和Moto方式两者编码的不同,两者都适用于Http协议传输,在1Mb下文本编辑方式为
///最佳选择,Moto适合大文件传输,在跨平台上无疑这是最好的选择
CompareMessageSize();
CompareMessageSize();
CompareMessageSize();
CompareMessageSize();
CompareMessageSize(); Console.ReadLine();
看运行结果:
显然在字节大于2000字节的时候MOTO方式的编码长度小于普通编码方式,而在小于2000字节的时候普通文本编辑要优于MTOM编码方式
所以在我们定义Binding的时候我们要分情况对待。
我们在wcf中应用MTOM的方式是通过定义Binding的方式进行的,看配置代码
<!--定义bangings-->
<bindings>
<!--文本编码方式是Mtom方式,传输协议是wsHttpBinding-->
<wsHttpBinding>
<binding name="wsHttpBing_IUpload" messageEncoding="Mtom" />
</wsHttpBinding> <!--流编码方式transferMode启动流模式,设置流模式类型-->
<basicHttpBinding>
<binding name="myBinding" transferMode="Streamed" />
</basicHttpBinding>
</bindings>
这里面定义了两种Binding编码方式,一个是采取Mtom方式、另一个是采取流媒体;其实两者是有区别的:
1、BasicHttpBinding发送的是明文数据,而WsHttpBinding发送的是加密和更加安全的数据
2、如果你希望有向后兼容的能力,并且支持更多的客户端,你可以选择basicHttpBinding,如果你确定你的客户端使用的 是.NET .0甚至更高的话,你可以选择wsHttpBinding
也就是说 basicHttpBinding扩展性好,而wshttpBinding必须给予..0以上版本 当然流媒体传输的速度相对会优于文本传输,但流开启的条件是有约束的:
有于stream的限制,只能对流模式使用传输级别的安全选项,并且无法打开 可靠会话,因此,流模式仅在下列系统定义的绑定中使用
BasicHttpBinding
NetTcpBinding
NetNamedPipeBinding
并且流模式传输时只能定义一个参数,如果添加多个参数,则自动转换成文本编码方式,用缓存来实现
不同的应用环境分情况对待,我们看上面的配置我们如何应用,客户端定数据契约和方法:
namespace service
{
[ServiceContract(Namespace = "http://localhost/Server")]
public interface IUplod
{
[OperationContract]
int Upload(Stream data);
}
public class UploadService : IUplod
{
public int Upload(Stream data)
{
int size = ;
int bytesRead = ;
byte[] buffer = new byte[]; //新建字节数组 //从流中读取信息存储到字节数组中
do
{
bytesRead = data.Read(buffer, , buffer.Length); //
size += bytesRead; }
while (bytesRead > );
data.Close();
return size;
}
}
}
采用自托管的方式:
namespace service
{
class Program
{
static void Main(string[] args)
{
using (ServiceHost host = new ServiceHost(typeof(UploadService)))
{
host.Opened += delegate
{
Console.WriteLine("服务已经启动");
};
host.Open();
Console.ReadLine();
}
}
}
}
客户端实现流传输:
byte[] binaryData = new byte[];
//定义1000长度的字节数组
MemoryStream sream = new MemoryStream(binaryData);
//生成内存里流.MemorySteam继承自sream,是stream的实例
ServiceUplod.UplodClient client = new ServiceUplod.UplodClient();
Console.WriteLine(client.Upload(sream));
//调用服务传入1000个字节
sream.Close();//关闭流
运行结果:
呵呵...这种方式只应用于http协议传输,通过MTOM或者流媒体传输实现,流媒体的传输,在使用TCP协议的情况下我们能更大限度的使用流媒体传输,当然既然优越就有它的限制性,下一篇我们分析,基于此种协议进行大文件传输
来自:http://www.cnblogs.com/zhijianliutang/archive/2011/11/28/2265858.html
转:wcf大文件传输解决之道(1)的更多相关文章
- 转:wcf大文件传输解决之道(2)
此篇文章主要是基于http协议应用于大文件传输中的应用,现在我们先解析下wcf中编码器的定义,编码器实现了类的编码,并负责将Message内存中消息转变为网络发送的字节流或者字节缓冲区(对于发送方而言 ...
- WCF大文件传输【转】
http://www.cnblogs.com/happygx/archive/2013/10/29/3393973.html WCF大文件传输 WCF传输文件的时候可以设置每次文件的传输大小,如果是小 ...
- WCF大文件传输服务
由于项目需要,自己写一个基于WCF的大文件传输服务雏形.觉得有一定的参考价值,因此放在网上分享. 目前版本为v1.1特点如下: 1.文件传输端口为18650 2.上传和下载文件 3.支持获取文件传输状 ...
- WCF大文件传输
WCF传输文件的时候可以设置每次文件的传输大小,如果是小文件的时候,可以很方便的将文件传递到服务端,但是如果文件比较大的话,就不可取了 遇到大文件的话可以采取分段传输的方式进行文件传输 思路: 1.客 ...
- WCF 大文件传输配置
<bindings> <webHttpBinding> <!--这个是接收大数据加的,设置WCF服务器端数据接收上限参数,此处单位字节,故2147483647字节==2G ...
- Nginx集群之WCF大文件上传及下载(支持6G传输)
目录 1 大概思路... 1 2 Nginx集群之WCF大文件上传及下载... 1 3 BasicHttpBinding相关配置解析... 2 4 编写 ...
- 大文件传输 分片上传 上传id 分片号 授权给第三方上传
https://www.zhihu.com/question/39593108 作者:ZeroOne链接:https://www.zhihu.com/question/39593108/answer/ ...
- 【转】Windows2008上传大文件的解决方法(iis7解决上传大容量文件)
2008上传大文件的解决方法:http://wenku.it168.com/d_000091739.shtml 2003上传大文件的解决方法:http://tech.v01.cn/windowsxit ...
- 利用Socket进行大文件传输
分类: WINDOWS 最近接触到利用socket进行大文件传输的技术,有些心得,与大家分享.首先看看这个过程是怎么进行的(如下图): 所以,我们需要三个socket在窗体加载的时候初始化: ...
随机推荐
- (1)DBA查询:数据库
1.数据库状态:[1]sys.databases [2]exec sp_spaceused 2.数据文件状态:[1]sys.master_files [2]查看ldf与mdf:sp_helpfil ...
- 文件压缩:zip
[root@localhost ~]# yum install -y zip unzip // 安装 zip 和 unzip [root@localhost ~]# ..txt // 压缩文件,要同时 ...
- pymysql 模块 使用目录
mysql python pymysql模块 基本使用 mysql python pymysql模块 增删改查 插入数据 介绍 commit() execute() executemany() 函数 ...
- python用%来处理字符串
%s 可以字符串拼接 msg='i am %s my hobby is %s' % ('lhf','alex') print(msg) 执行结果: i am lhf my hobby is alex ...
- php循环方法实现先序、中序、后序遍历二叉树
二叉树是每个节点最多有两个子树的树结构.通常子树被称作“左子树”(left subtree)和“右子树”(right subtree). <?php class Node { public $v ...
- spark提交任务的三种的方法
在学习Spark过程中,资料中介绍的提交Spark Job的方式主要有三种: 第一种: 通过命令行的方式提交Job,使用spark 自带的spark-submit工具提交,官网和大多数参考资料都是已这 ...
- hash值重写,就是以自己定义的规则来显示hash值
未重写hashCode值 重写hashCode后的值
- eclipse中Tomcat服务器缓存位置,以及清理Tomcat缓存
在Eclipse中进行Web开发,一般都会将项目直接在Eclipse中的Tomcat服务器运行,有时候修改了程序和页面之后,运行结果还是原来的 tomcat服务器中缓存的程序或者页面,需要清理缓存之后 ...
- 7个Java项目,或许你的大学老师就会布置
前言: 有天吃饭和朋友聊天,说到大学老师布置的开发项目,结果我们一干人说出来的都基本一样,入门级别的计算器啦,稍微大一点的记事本啦,然后到后面的图书管理系统啊,购物网站啊-- 发现这些项目都是大学老师 ...
- node加密解密 crytpo
var crypto = require('crypto'); exports.encrypt = function (str, secret) { var cipher = crypto.creat ...