WCF后传系列(6):消息如何传递之绑定Part 1

概述

每一个服务终结点都包含一个地址Address、一个绑定Binding 和一个契约Contract。契约指定可用的操做,绑定指定如何与服务进行通讯,而地址指定查×××的位置,在WCF专题系列前5篇中,深刻了解了WCF中寻址的细节;从本文开始,将深刻了解经典“ABC”中的B,即绑定Binding,了解它的原理,如何使用绑定通讯以及在元数据中如何公开

什么是绑定

从本质上说,WCF是一个通讯服务框架,它容许咱们使用不一样的传输协议(如Http、Https、TCP、P2P、IPC和MSMQ等),使用不一样的消息编码形式(文本、二进制和MTOM),跟不一样的WS-*系列规范(如WS-Security、WS-Atomic Transaction等)交互。全部这些细节都是由通道堆栈来处理的,看一下Aaron Skonnard给出的这幅图:
TerryLee_WCF_19
图1
最底层传输组件读入消息,消息编码器将传入字节读取为逻辑消息对象,以后消息经过通道堆栈中的各个消息,它们执行各自的处理。若是对这三者之间之间进行组合,至少能够获得上千种不一样的通讯方式,可是这些传输、编码以及协议之间有些是互斥的,有些相互约束,也就是说,对于开发人员来讲,每构建一个服务,都要须要考虑这三者之间是否能够共存,是不是高效的通讯方式,显然这个工做是很是复杂的,要求开发者必须了解全部的传输、编码以及协议等。
为了简化这三者之间的管理,WCF中引入了绑定的概念(Binding),每一个绑定中都封装了传输协议、消息编码和多种协议的集合,这样在构建服务时,咱们就能够直接选择一个适用的绑定,经过调整它们的属性来适应需求,若是系统内置的绑定经过调整属性仍然不能达到咱们的要求,这时才会考虑自定义绑定。这样,经过绑定,就把应用程序编程模型与通道模型(后面会有专门的文章写到)关联了起来,对于开发者来讲,就无需再考虑复杂的底层通道模型,直接使用应用程序编程模型。

绑定元素

在WCF中,绑定由绑定元素组成,每一个绑定元素用来描述终结点与客户端通讯方式中的某个方面,绑定元素继承于BindingElement,其中最重要的绑定元素有以下三种:
1.编码绑定元素(Encoding Binding Element):如采用文本、二进制仍是MTOM的方式;
2.传输绑定元素(Transport Binding Element):使用Http、TCP或者MSMQ进行传输;
3.协议绑定元素(Protocol Binding Element):指定可靠性、安全性、事务等。
每一个绑定必需要有一个编码绑定元素和一个传输绑定元素,以及包括任意数目的其余协议绑定元素。

通道模型中使用绑定

在WCF中,提供了两个层面的模型,一是针对开发人员的应用程序编程模型;二是用于通讯的通道模型(后面会专门讲到)。在通道模型编程中,服务端侦听并接收消息的第一步就是建立绑定,以下面的代码:
// 建立自定义绑定
BindingElement[] bindingElements = new BindingElement[2];
bindingElements[0] = new TextMessageEncodingBindingElement();
bindingElements[1] = new HttpTransportBindingElement();
CustomBinding binding = new CustomBinding(bindingElements);
这里建立了一个自定义绑定,并在其中加入了两个必须的绑定元素,消息编码采用文本的方式,而传输绑定元素采用了内置的HttpTransportBindingElement。接下来就可使用自定义的绑定来建立通道侦听器,并进行消息的侦听,以下代码所示:
/// <summary>
/// Author:TerryLee
/// Url:[url]http://www.cnblogs.com/terrylee[/url]
/// </summary>
/// <param name="args"></param>
static void Main(string[] args)
{
    // 建立自定义绑定
    BindingElement[] bindingElements = new BindingElement[2];
    bindingElements[0] = new TextMessageEncodingBindingElement();
    bindingElements[1] = new HttpTransportBindingElement();
    CustomBinding binding = new CustomBinding(bindingElements);
    // 使用自定义绑定建立通道侦听器         
    IChannelListener<IReplyChannel> listener =
          binding.BuildChannelListener<IReplyChannel>(
             new Uri("http://localhost:8080/ChannelApp"),
             new BindingParameterCollection());
    // 监听消息
    listener.Open();
    Console.WriteLine("Listening for incoming channel connections");
    IReplyChannel channel = listener.AcceptChannel();
    Console.WriteLine("Channel accepted. Listening for messages");
    channel.Open();
    while (true)
    {
        RequestContext request = channel.ReceiveRequest();
        // 读取请求的消息
        Message message = request.RequestMessage;
        Console.WriteLine("Message Received");
        Console.WriteLine("Message Action: {0}", message.Headers.Action);
        string body = message.GetBody<string>();
        Console.WriteLine("Message Content: {0}", body);
        // 发送响应消息
        Message replymessage = Message.CreateMessage(
            binding.MessageVersion,
            "http://www.cnblogs.com/terrylee",
             body);
        request.Reply(replymessage);
        // 释放对象
        message.Close();
        request.Close();
        channel.Close();
        listener.Close();
    }
}
其中的代码咱们就再也不解释,到关于WCF中通道编程模型一文中,还会进行详细的讲解。

自定义系统绑定

在WCF中,已经内置了大量的绑定供咱们使用,可是若是这些绑定不能知足实际的开发需求,咱们能够经过几种办法来对系统内置绑定进行自定义,一是经过配置文件进行绑定的配置,以下代码所示,配置BasicHttpBinding的消息编码为MTOM,而安全性为消息级别:
<services>
  <service name="TerryLee.WCFAddressing.Service.CalculatorService"
           behaviorConfiguration="calculatorBehavior">
    <host>
      <baseAddresses>
        <add baseAddress="[url]http://localhost:8887/Calculator[/url]"/>
      </baseAddresses>
    </host>
    <endpoint address=""
              binding ="basicHttpBinding"
              contract="TerryLee.WCFAddressing.Contract.ICalculator"
              name="defaultBinding"
              bindingConfiguration="myBindingConfiguration">
    </endpoint>
  </service>
</services>
<bindings>
  <basicHttpBinding>
    <binding name="myBindingConfiguration"
             messageEncoding="Mtom">
      <security mode="Message"></security>
    </binding>
  </basicHttpBinding>
</bindings>
二是能够经过编码的方式,以下代码所示:
BasicHttpBinding binding = new BasicHttpBinding();
binding.Security.Message.ClientCredentialType = 
    BasicHttpMessageCredentialType.Certificate;
binding.Security.Mode = BasicHttpSecurityMode.Message;
binding.MessageEncoding = WSMessageEncoding.Mtom;
考虑到程序部署以后的修改,仍是推荐使用配置的方式。同时,咱们彻底能够利用系统内置绑定建立一个自定义的绑定,或者咱们在自定义绑定过程当中,能够直接经过已有的系统内置绑定来建立一个绑定元素集合,以下代码所示:
BasicHttpBinding binding = new BasicHttpBinding();
binding.Security.Message.ClientCredentialType = 
    BasicHttpMessageCredentialType.Certificate;
binding.Security.Mode = BasicHttpSecurityMode.Message;
binding.MessageEncoding = WSMessageEncoding.Mtom;
CustomBinding mybinding = new CustomBinding(binding);
BindingElementCollection myElements =
    binding.CreateBindingElements();
在下篇文章中,我将会介绍如何彻底从新进行自定义绑定。

元数据中公开绑定

在WCF中,通讯的双方应该就通讯的细节达成一致,既然绑定中封装了全部通讯的细节,而服务端惟一公开这些细节的方式就是元数据,因此绑定的相关信息应该经过公开WSDL和 WS-Policy 形式的元数据,这样服务就能够和客户端共享绑定配置,以BasicHttpBinding为例,它所使用的传输是Http,而消息编码采用文本的方式,若有下面这样一端配置:
<services>
  <service name="TerryLee.WCFAddressing.Service.CalculatorService"
           behaviorConfiguration="calculatorBehavior">
    <host>
      <baseAddresses>
        <add baseAddress="[url]http://localhost:8887/Calculator[/url]"/>
      </baseAddresses>
    </host>
    <endpoint address="[url]http://localhost:8887/CalculatorService[/url]"
              binding ="basicHttpBinding"
              contract="TerryLee.WCFAddressing.Contract.ICalculator"
              name="defaultBinding"
              bindingConfiguration="myBindingConfiguration">
    </endpoint>
  </service>
</services>
<bindings>
  <basicHttpBinding>
    <binding name="myBindingConfiguration"
             messageEncoding="Mtom">
    </binding>
  </basicHttpBinding>
</bindings>
经过绑定配置,设置BasicHttpBinding的消息编码为MTOM,在元数据中,能够看到采用Http进行传输:
<wsdl:binding name="defaultBinding" type="tns:ICalculator">
  <wsp:PolicyReference URI="#defaultBinding_policy" />
  <soap:binding transport="[url]http://schemas.xmlsoap.org/soap/http[/url]" />
  <wsdl:operation name="Add">
    <soap:operation soapAction="[url]http://tempuri.org/ICalculator/Add[/url]"
                    style="document" />
    <wsdl:input>
      <soap:body use="literal" />
    </wsdl:input>
    <wsdl:output>
      <soap:body use="literal" />
    </wsdl:output>
  </wsdl:operation>
</wsdl:binding>
对于消息编码MTOM,放在Policy的配置中,以下代码所示:
<wsp:Policy wsu:Id="defaultBinding_policy">
  <wsp:ExactlyOne>
    <wsp:All>
      <wsoma:OptimizedMimeSerialization
        xmlns:wsoma="[url]http://schemas.xmlsoap.org/ws/2004/09/policy/optimizedmimeserialization[/url]" />
    </wsp:All>
  </wsp:ExactlyOne>
</wsp:Policy>

0javascript

收藏css

lihuijun

203篇文章,74W+人气,0粉丝

Ctrl+Enter 发布html

发布java

取消jquery

扫一扫,领取大礼包linux

0git

分享
lihuijun
相关文章
相关标签/搜索