初识WCF6
参考:http://blog.csdn.net/songyefei/article/details/7397296
元数据交换
通过前两篇的学习,我们了解了WCF通信的一些基本原理,我们知道,WCF服务端和客户端通过共享元数据(包括服务协定、服务器终结点信息)在两个终结点上建立通道从而进行通信。我们通过手写代码(或配置)的方式为服务端编写了元数据信息,没有借助元数据交换就实现了通信。然而在实际应用中,元数据往往是很多的,而且重复编写元数据的工作也是不值得的,因此必然会用到元数据交换的方式让客户端获取元数据,本篇我们就来进一步了解一下元数据和元数据交换。
1. 元数据是怎样提供的
我们知道,元数据包括了要和服务端进行通信的所有信息,包括服务协定接口、服务端终结点地址、绑定等信息,它指出了客户端应该到何处去寻找服务以及怎样调用服务的一切线索。但是服务端是怎样公布其元数据的呢?
答案是使用WSDL文件,WSDL即Web Service Description Language,Web服务描述语言,它是一个XML文件,在这个文件中按照一定的标准来对Web Service进行描述,他是符合W3C标准的,因为WCF是被设计为供不同平台调用的服务框架,所以客户端可能是非微软平台的,比如Java什么的。因此WCF必须使用WSDL这种国际标准的描述方法来描述服务才能被众多的平台所访问。
2. 元数据交换的过程是怎样的
在WCF服务端的运行时,有一组类库随时待命把服务的元数据输出为WSDL描述提供给请求者,只要有客户端按照服务端约定的方法来请求元数据,服务端立即将服务运行时状态写成WSDL文件提供。客户端得到的实际上就是WSDL文件(还有一些框架描述文件XSD),客户端拿到文件后再使用自己的方法来解读WSDL,把他翻译成客户端可用的源代码或配置文件,这时客户端就得到了服务的编程模型,通过一些代理类,客户端甚至可以像调用本地对象一样使用WCF服务。
因此整个过程是这样:客户端向服务端请求元数据交换-->服务端运行时将元数据编写成WSDL文件提供-->客户端获得文件-->客户端翻译文件-->客户端根据翻译结果生成本地类代码和配置-->客户端获得服务的本地编程模型。这就是元数据交换的过程。
3. 获得WSDL
在微软平台中,有两种方法来进行元数据交换,第一是使用服务引用,第二是使用元数据实用工具(svcutil.exe)来进行,我们先学习这个工具。
这个工具可以在Windows SDK中找到,具体位置为 C:\Program Files\Microsoft SDKs\Windows\v6.0\Bin,如果你有VS2010,可以启动VS2010的命令行工具,这样就可以在任何目录下使用这个程序。
我们先看一个例子,就是我们在前几篇中建立的IIS服务HelloWCFService,它被我寄宿在IIS中。
源代码如下(HelloWCF.cs):
using System;
using System.ServiceModel; namespace LearnWCF
{
[ServiceContract]
public interface IHelloWCF
{
[OperationContract]
string HelloWCF();
} public class HelloWCFService : IHelloWCF
{
public string HelloWCF()
{
return "Hello WCF!";
}
}
}
配置文件(web.config)如下:
<configuration>
<system.serviceModel>
<services>
<service name="LearnWCF.HelloWCFService" behaviorConfiguration="metadataExchange">
<endpoint address="" binding="wsHttpBinding" contract="LearnWCF.IHelloWCF"/>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="metadataExchange">
<serviceMetadata httpGetEnabled="true"/>
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
</configuration>
在浏览器中输入服务地址会如下图所示:

看到系统提示的那行命令了么?系统在告诉我们如何使用svcutil.exe来获得元数据。我们现在试一下,首先打开VS2010命令行:
开始-->所有程序-->Visual Studio 2010-->Visual Studio Tools-->Visual Studio命令行提示


我们导航到一个目录下准备获得元数据文件。
我们暂时不按照浏览器提供给我们的方法做,因为按照那个方法做就把获得WSDL和翻译WSDL为客户端代码合在一起了,我们先获得WSDL元数据文件,看看它是什么样子的。我们按如下的指令做:
svcutil.exe /t:metadata http://localhost/iisservice/hellowcfservice.svc?wsdl

可以看到生成了3个文件,包括两个架构文件和一个WSDL文件,这些就是服务端元数据的描述了,所有的客户端请求到的实际上都是这个文件。WSDL的规范比较多,关于它的内容,我们今后再展开来看,不过简单的打开看一下就能看出一些与服务协定、绑定、操作这些东西相关的地方。
4. 翻译WSDL文件
WSDL是一个XML文件,其实就是个文本文件,客户端必须将其按照自己的平台特点把他翻译成本地代码文件来使用。svcutil当然会提供这个功能。在wsdl文件所在目录下使用如下的命令就可以把WSDL文件翻译成本地代码文件:
svcutil *.wsdl *.xsd
顾名思义,就是根据当前目录下的所有的WSDL文件和XSD文件来生成客户端代码文件。过程会是这样

可以看到,生成了一个cs文件和一个配置文件,这些就是根据WSDL文件翻译成的客户端代码文件了。打开来看看,一定不陌生,就是使用ClientBase<>来生成一个客户端代理类并把终结点的信息配置在了.config文件里。把这两个文件包含在客户端的项目中并把output.config改成app.config就可以了。
4. 更好地使用元数据交换工具
之前我们了解了使用svcuitl.exe来获取WSDL并翻译成客户端代码的过程。实际上这两步可以合二为一。直接执行下面的命令可以直接获得客户端文件:
svcutil.exe http://localhost/iisservice/hellowcfservice.svc?wsdl
这样它就不会生成WSDL而直接生成客户端文件了。
不过按照这样的方式生成的文件可能不太符合我们的要求,我们可以加上一些参数来指定我们输出的文件名:
svctuil.exe /out:ClientProxy.cs /config:app.config http://localhost/iisservice/hellowcfservice.svc?wsdl
这样输出的文件我们就可以直接包含在客户端项目中使用了。
5. 使用服务引用
其实使用服务引用跟使用svcutil.exe生成的客户端模型是一样的,不过服务引用保留了WSDL文件(以及一些相关的七七八八的文件),没有svcutil.exe来得那么清爽,但是它跟VS2010集成,使用起来很简单,而且当服务发生变化时,只需要右击服务引用选择更新服务就可以重新下载WSDL了
6. 展开一点点
作为服务端,公开元数据是需要配置的,不同的配置会导致元数据公开的方式不同。
我们要记住,WCF服务端公开元数据必须具备两个条件:
(1) 为服务添加ServiceMetadata行为。
(2) 打开元数据交换终结点。
二者缺一不可。
WCF的公开元数据的手段主要有两种:
第一种:通过HTTP GET方法。
这就是在前文中我们看到的方法,我们可以使用HTTP Get的方法来获得WSDL文件即在服务地址.svc后面跟上?wsdl的方法直接请求到WSDL文件。我们可以直接在浏览器中输入服务端地址.svc?wsdl,浏览器就直接获得了WSDL文件并为我们显示出来了。

还有相应的框架描述文件(XSD)

如果想采用这种元数据公开方式,必须配置服务的ServiceMetadata行为,并指定httpGetEnabled = "true",而元数据公开终结点不必配置,系统会自动配置一个,配置文件的写法如下:
<configuration>
<system.serviceModel>
<services>
<service name="LearnWCF.HelloWCFService" behaviorConfiguration="metadataExchange">
<endpoint address="" binding="wsHttpBinding" contract="LearnWCF.IHelloWCF"/>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="metadataExchange">
<serviceMetadata httpGetEnabled="true"/>
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
</configuration>
在这种配置下,访问元数据的方法是访问下面的地址:
http://localhost/iisservice/hellowcfService.svc?wsdl
第二种:通过MEX元数据交换终结点。
在这种方式下,我们首先要保证服务拥有ServiceMetadata行为,但是httpGetEnabled可以不必为true。此外我们还需要为服务显式地添加一个终结点,这个终结点的地址、绑定和协定都是指定的我们不能更改
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
配置文件的写法如下:
<configuration>
<system.serviceModel>
<services>
<service name="LearnWCF.HelloWCFService" behaviorConfiguration="metadataExchange">
<endpoint address="" binding="wsHttpBinding" contract="LearnWCF.IHelloWCF"/>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange"/>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="metadataExchange">
<serviceMetadata />
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
</configuration>
如果按这种配置,我们必须按照如下地址来访问公开的元数据:
http://localhost/iisservice/hellowcfservice.svc/mex
注意,由于没有开启HTTP GET,我们不能在浏览器中直接输入这个地址来获取WSDL了(会提示400错误),我们必须通过svcutil.exe或添加服务引用的方式来访问。
使用svcutil.exe或服务引用的时候可以不关心元数据公开方式是HTTP GET还是Mex,他们会自动寻找到合适的方式,只需要把服务的svc文件地址输入就可以了,但是我们应该知道,这两种元数据公开的方式是有区别的。
6. 总结
通过今天的学习,我们进一步了解了WCF元数据的和元数据交换的原理。虽然我们在实际工程中都会并且应该使用元数据交换工具来帮助提高效率,但是这背后发生的所有环节也是我们应该掌握的。
相关资源
MSDN关于Svcutil.exe用法的文档
http://msdn.microsoft.com/zh-cn/library/aa347733.aspx
初识WCF6的更多相关文章
- Android动画效果之初识Property Animation(属性动画)
前言: 前面两篇介绍了Android的Tween Animation(补间动画) Android动画效果之Tween Animation(补间动画).Frame Animation(逐帧动画)Andr ...
- 初识Hadoop
第一部分: 初识Hadoop 一. 谁说大象不能跳舞 业务数据越来越多,用关系型数据库来存储和处理数据越来越感觉吃力,一个查询或者一个导出,要执行很长 ...
- python学习笔记(基础四:模块初识、pyc和PyCodeObject是什么)
一.模块初识(一) 模块,也叫库.库有标准库第三方库. 注意事项:文件名不能和导入的模块名相同 1. sys模块 import sys print(sys.path) #打印环境变量 print(sy ...
- 初识IOS,Label控件的应用。
初识IOS,Label控件的应用. // // ViewController.m // Gua.test // // Created by 郭美男 on 16/5/31. // Copyright © ...
- UI篇(初识君面)
我们的APP要想吸引用户,就要把UI(脸蛋)搞漂亮一点.毕竟好的外貌是增进人际关系的第一步,我们程序员看到一个APP时,第一眼就是看这个软件的功能,不去关心界面是否漂亮,看到好的程序会说"我 ...
- Python导出Excel为Lua/Json/Xml实例教程(一):初识Python
Python导出Excel为Lua/Json/Xml实例教程(一):初识Python 相关链接: Python导出Excel为Lua/Json/Xml实例教程(一):初识Python Python导出 ...
- 初识SpringMvc
初识SpringMvc springMvc简介:SpringMVC也叫Spring Web mvc,属于表现层的框架.Spring MVC是Spring框架的一部分,是在Spring3.0后发布的 s ...
- 初识redis数据类型
初识redis数据类型 1.String(字符串) string是redis最基本的类型,一个key对应一个value. string类型是二进制安全的.意思是redis的string可以包含任何数据 ...
- Redis初识、设计思想与一些学习资源推荐
一.Redis简介 1.什么是Redis Redis 是一个开源的使用ANSI C 语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value 数据库,并提供多种语言的API.从2010 年 ...
随机推荐
- 【文档】二、Binlog结构和内容概述
binlog是一系列文件,这些文件包含了Mysql服务实例中数据的变化. binlog包含一系列二进制日志文件,还包含一个索引文件. 每个日志文件包含了一个4字节的魔法数,后面跟着描述数据变化的事件内 ...
- hibernate配置hbm2ddl.auto的四个参数
<!-- Drop and re-create the database schema on startup --> <!-- hbm(hibernatemapping) ,ddl( ...
- 实现基于NTP协议的网络校时功能
无论PC端还是移动端系统都自带时间同步功能,基于的都是NTP协议,这里使用C#来实现基于NTP协议的网络校时功能(也就是实现时间同步). 1.NTP原理 NTP[Network Time Protoc ...
- VisualSVN Server提供程序无法执行所尝试的操作 0x80041024
VisualSVN安装后没有提供VisualSVN Server Manager的快捷方式,如下图: 可以在安装目录的bin文件夹下找到VisualSVN Server.msc,添加快捷方式.建议Vi ...
- 对Tensorflow中tensor的理解
Tensor即张量,在tensorflow中所有的数据都通过张量流来传输,在看代码的时候,对张量的概念很不解,很容易和矩阵弄混,今天晚上查了点资料,并深入了解了一下,简单总结一下什么是张量的阶,以及张 ...
- http协议--文章一
一 原理区别 一般在浏览器中输入网址访问资源都是通过GET方式:在FORM提交中,可以通过Method 指定提交方式为GET或者POST,默认为GET提交 Http定义了与服务器交互的不同方法,最基本 ...
- centOS 7下无法启动网络(service network start)错误解决办法(应该是最全的了。。。)
今天在centOS 7下更改完静态ip后发现network服务重启不了,翻遍了网络,尝试了各种方法,终于解决了. 现把各种解决方法归纳整理,希望能让后面的同学少走点歪路... 首先看问题:执行serv ...
- Python基础(10) - 异常
Python 异常:程序出现了错误而在正常控制流以外采取的行为 Python中常见的异常: 1. NameError:尝试访问一个未声明的变量 >>> something Trace ...
- 微信小程序随笔。笔记本忘家里了
所有ui组件都有的共有属性id.class.style.hidden.data-*.bind* / catch*.
- 五、curator recipes之选举主节点Leader Latch
简介 在分布式计算中,主节点选举是为了把某个进程作为主节点来控制其它节点的过程.在选举结束之前,我们不知道哪个节点会成为主节点.curator对于主节点选举有两种实现方式,本文示例演示Latch的实现 ...