WCF(一) Windows Communication Foundation 概述
WCF被推出来很久了,但是对于刚刚接触WCF的人们来说,想必会有以下问题:
WCF是什么?为什么我们需要WCF?WCF能给我们带来什么?学习WCF的难易度如何?
以上这些问题的答案在网上已经很多介绍了,我在此本不必再浪费口舌,只是大概提一下。
WCF: Windows Communication Foundation.从名字可以看出,这是一项和通信密切相关的技术。我们知道软件设计到处是通信。当然我们必须学了以后,才能更好的知道它、运用它,而不能单单从它的名字揣测它的威力如何。
WCF提供给了我们一种面向服务(SOA)的程序设计的解决方案:何为面向服务?之前有使用过.NET Web Service 开发的朋友,很容易理解面向服务是咋回事。.Net Web Service也是更早的时候微软提出的一种面向服务的解决方案。面向服务是一种标准,不同的公司可以给出各自的实现框架。单纯的讲SOA,毕竟太抽象了。学习一下SOA的一种实现比如.Net Web Service 或WCF以后,相信不用任何文字定义,你也会深刻地理解SOA。所以,何为面向服务?这里不解释,等你学一段时间的WCF,你自己就有答案了,真的,呵呵。
WCF提供了统一的通信模型:以前我们写通信可以是:TCP/UDP,Socket等这些方式,在WCF中被统一了。至于如何统一?这里先不说了。所以,当你在项目中同时使用了多种通信方式时,可以考虑使用WCF了。
以前没有接触过Web Service 开发接受WCF也不是难事,相信我!
这次,我们只做一点WCF的概述,更为详细的知识到后面再慢慢展开。边学习,边交流。
1. 打开VS 2010 -->file-->new Project ,选择C#下面的WCF,选择WCF Library。如下图:
2.此时有了IService 文件和一个Service 文件。Service文件实现的IService中的两个函数:GetData函数和GetDataUsingDataContract函数。至于这两个函数是干嘛的?想必不用说大家都能看懂的。之后详细说明他们的角色。这里也不修改类名和函数名了,方面起见。
3.按F5运行程序,此时打开了一个WCF 的一个测试 窗口。如下图:先测试GetData函数
测试结果:
同样的方法测试GetDataUsingDataContract函数:结果如下图
4.上面完成了对WCF的函数功能的简单测试,能说明的是Service正确地实现了IService。接下来我们需要将这个WCF部署到一个Web Site中。
在解决方案上面右击:New -->New Web Site 如下图:
修改web site项目中的Service.svc 文件里面的 Service="WcfFirstDemoServiceLib.Service1" (其中意思就是已命名空间加类名的方式指定service的名字)
5.添加一个winForm程序 ,用它来调用WCF的函数。
6.给winForm 项目添加web 引用:
7.在Form1上面添加几个控件,然后再在Form1.cs中添加下列几行代码:
Form1.cs
8.Ctrl+F5 运行:
说明:这里只是概述WCF,所以全部在使用WCF的默认设置,也没有详细说明每一步的作用是什么。只是给出三个项目:项目1是WCF的库也是WCF功能核心。项目2是WCF的host用于向外界发布WCF提供的接口(注:接口这里指提供了哪些函数可被调用)。项目3是WCF的客户端用于调用WCF。其实,这里已经隐约体现出一点点分布式软件架构的意思了:有专门提供功能实现的模块,有专门提供对外公开接口的模块,有专门作为调用的模块。
后面详细WCF每一部分的。欢迎关注!!
WCF(二) Contract
上篇只是介绍了WCF的概述。具体的设置全是使用默认,这当然不可能满足我们开发的需要。如果仔细理一理的话,你会发现WCF里面的设置其实不算多的(与SharePoint比较的话)。从这篇开始,我们一点一点来展开学习。这次先提最最常用的Contract。
Contract有人翻译为:协定,契约。
WCF中有四种contract: 分别是:1.Service Contract. 2.Data Contract. 3. Fault Contract. 4.Message Contract.
还拿上篇中的例子(其实就是有Visual Studio2010替我们默认生成的代码了)说事:
1.Service Contract共分为两个部分:定义部分和实现部分。
[ServiceContract]
2 public interface IService1
3 {
4 [OperationContract]
5 string GetData(int value);
6
7 [OperationContract]
8 CompositeType GetDataUsingDataContract(CompositeType composite);
9
10 }
这里IService就是一个service contract的定义部分,指定了哪些函数(Operation)向外公开(即:可供Client调用)。被[OperationContract]修饰的函数可以被客户端(Client)调用。
public class Service1 : IService1
2 {
3 public string GetData(int value)
4 {
5 return string.Format("You entered: {0}", value);
6 }
7
8 public CompositeType GetDataUsingDataContract(CompositeType composite)
9 {
10 if (composite == null)
11 {
12 throw new ArgumentNullException("composite");
13 }
14 if (composite.BoolValue)
15 {
16 composite.StringValue += "Suffix";
17 }
18 return composite;
19 }
20 }
Service1实现了IService接口,是Service Contract 的实现部分。VS自动生成的代码太简单了,不解释了。这里为了尽可能简单地说明问题。具体项目中,你可能需要连接数据库增、删、改、查数据,或对数据按n多复杂的业务规则进行数据处理以后,返回给client,等。[ServiceContract] Attribute 可以有以下Property 的:
[OperationContract] Attribute 可以有以下Property 的:
Action 对请求设置WS-Addressing 的action
AsynchPattern 异步模式
HasProtectionLevel 消息是否加密,签名
IsInitiating 表明该函数被调用开始时是否要在server上面初始化一个session
IsOneWay 表明函数被client调用以后,client是否会等待函数返回
IsTerminating 表明该函数被调用结束时是否要在server上面关闭session
Name 设置函数的名字,在client端可见的名字,默认就是函数名字
ProtectionLevel
ReplyAction 设置函数返回消息的SOAP action
1.1 我们修改一下Service的默认命名空间,使他更make scene(有意义).
打开上篇中的解决方案WcfFirstDemo.sln
右键WebHost项目下面的文件:Service.svc,如下图:
点击连接如图:
得到如图效果:
默认Namespace是http://tempuri.org/
微软官方建议:修改Service的Namespace,使其包含:公司域名+项目名+版本号(如:日期表示版本号)
修改项目:WcfFirstDemoServiceLib下面的IService.cs代码如下:
[ServiceContract(Namespace="http://wwww.cnblogs.com/WCF/2012/07/28")]
2 public interface IService1
3 {
4 [OperationContract]
5 string GetData(int value);
6
7 [OperationContract]
8 CompositeType GetDataUsingDataContract(CompositeType composite);
9
10 }
右键项目WcfFirstDemoServiceLib 选择重新编译,成功以后,重新用浏览器打开service.svc,得到如下图:
此时已经改变了Service默认的Namespace了,client端需要更新一下,否则运行client端是会报异常的.操作如下图:
1.2修改Service的Name.默认情况下定义service的接口部分(如本例:IService)的名字就是Service的名字.但有时需要让client看到的service的名字跟server端看到的的service名字不一样.
我们先看一下WCFClient项目下面的app.config
修改WcfFirstDemoServiceLib项目下的IService.cs文件
[ServiceContract(Namespace="http://wwww.cnblogs.com/WCF/2012/07/28",Name="DemoService")]
2 public interface IService1
3 {
4 [OperationContract(Name="GetAge")]
5 string GetData(int value);
6
7 [OperationContract]
8 CompositeType GetDataUsingDataContract(CompositeType composite);
9
10 }
重新编译WcfFirstDemoServiceLib项目.更新client端对service的引用.此时你会发现client端app.config文件前后发生了变化:
此时WCFClient所生成的代理类的名字也会变化,你需要修改WCFClient
DemoService proxy = null;
1 private void Form1_Load(object sender, EventArgs e)
2 {
3 proxy = new DemoServiceClient("BasicHttpBinding_DemoService");
4
5 }
1 private void btnGetData_Click(object sender, EventArgs e)
2 {
3 this.tbOutputBox.Text = proxy.GetAge(Convert.ToInt32(this.tbInputbox.Text));
4 }
注:上面代码中DemoService 对应server端IService接口;DemoServiceClient对应server端的Service类.
当client端的app.config文件中只包含一个endpoint时,可以直接用proxy=new DemoServiceClient()传空参数,当然也可以传这个endpoint的name进去;但是当app.config包含多个endpoint时必须把endpoint的name传进去才能new出proxy对象.
2.DataContract.
WCF的Server和client之间利用特定格式的Message(消息)进行通信的。但是我们使用的高级语言:不论在Server端还是在client端,我们处理的业务数据都被封装成了对象。所以想把server端的对象传送到client(或相反方向)时,我们必须有能力把这些对象转换成特定格式的message,到另一端接收到message后再把它转回成对象。C#的基础类型(如int,string,float,bool等)可以做到这一点。因为一旦确定了Server端或client所用的编程语言,就可以确定这些基础类型的内存占用情况(例如:一个Server端定义的int变量在C#中占用多大内存空间是已知的,那么client端即便使用的是其他语言也完全可以计算出该变量的内存占用情况)。可是我们自己定义的类型可是五花八门了(比如:server端定义的Person类new出的object到底占用多大内存空间,就算Server端与client端使用同一种语言,Client端是不知道的Person类的对象的内存占用情况的。)
遇到这种请款下,我们怎么办呢?这时就轮到[DataContract]露面了。
ServiceContract做的工作是指定service向client提供了哪些函数可供调用。DataContract做的工作就是指定在Server端与client端之间可以传送的数据。[DataContract]的作用就是指定当需要传送某个类(如:Person类)的对象时,将该对象转化成为XML.接受方接到XML以后,再按同样的方式还原成对象。
[DataContract]Attribute 标在class定义上面一行。[DataMember]Attribute标在Property定义上面一行,field不需要Attribute修饰。
[DataContract]
2 public class CompositeType
3 {
4 bool boolValue = true;
5 string stringValue = "Hello ";
6
7 [DataMember]
8 public bool BoolValue
9 {
10 get { return boolValue; }
11 set { boolValue = value; }
12 }
13
14 [DataMember]
15 public string StringValue
16 {
17 get { return stringValue; }
18 set { stringValue = value; }
19 }
20 }
[DataContract]Attribute 可以像[ServiceContract]一样设置Name和Namespace.
[DataMember] Attribute 可以有以下属性:
EmitDefaultValue 设置一个默认值
IsRequired 进行序列化/反序列化时该值一定不可为空值
Name Property的名字
Order 设置进行序列化/反序列化的顺序
2.1修改DataContract的默认namespace ,
[DataContract(Namespace = "http://wwww.cnblogs.com/WCF/2012/07/28", Name = "CompositeTypeDemo")]
2 public class CompositeType
3 {
4 bool boolValue = true;
5 string stringValue = "Hello ";
6
7 [DataMember(Name="GetBool")]
8 public bool BoolValue
9 {
10 get { return boolValue; }
11 set { boolValue = value; }
12 }
13
14 [DataMember(Name="GetString")]
15 public string StringValue
16 {
17 get { return stringValue; }
18 set { stringValue = value; }
19 }
20 }
这样在client端看到的类(class)名,函数名与server看到的就不相同.
3.Fault Contract.
在WCF中处理异常(Exception)的方法有些特殊.我们不能单从server处理exception,需要进一步将Exception从server传送到client. 在后面我们再单独讨论WCF的Exception处理.
4. Message Contract.
Message Contract与Data Contract都是作用在传送的对象(object)上面.
不同的是:
datacontract是将object序列化化为xml. 实现object的各个property与xml 文本中各个node(节点)的对应);
messagecontract将对象组装成message(指定Message的Header,body).实现的是object各个property与消息的各个元素的对应.
具体的深层区别于联系,以及使用场景,我还不懂,希望有朋友对着块比较懂的,可以交流一下.如果后面等我弄懂了的话,我再专门写博文交流.
WCF(三) Message pattern