第一步:创建一个空的解决方案,新建一个WCF服务应用程序项目(使用默认名字) 来模拟服务端,新建一个控制台应用程序项目(名称改为 ConsoleApp)来模拟客户端。

第二步:简单分析WcfService_1项目,该项目内容如下:

一句话总结:这个项目模拟服务器端,Service1.svc文件封装的就是提供给客户端的服务引用,Service1.svc.cs文件里是服务引用的具体实现。但这里因为Service1.svc.cs文件里的主要内容——Service1类是继承于 IService1.cs文件里的 IService1接口,所以重头戏分了一半给IService1.cs文件。

首先看IService1.cs文件,从名字上可以看得出这个是接口文件,里面定义了了一些接口,接口声明了一些方法。我在里面添加两个类 public class Student1 、 public class Student2 和一个方法 Student1 StudentWriteName(string name)用作测试。代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.ServiceModel.Web;
using System.Text; namespace WcfService_1
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的接口名“IService1”。
[ServiceContract]
public interface IService1
{ [OperationContract]
string GetData(int value); [OperationContract]
CompositeType GetDataUsingDataContract(CompositeType composite); // TODO: 在此添加您的服务操作 //新增方法
[OperationContract]
Student1 StudentWriteName(string name);
} // 使用下面示例中说明的数据约定将复合类型添加到服务操作。
[DataContract]
public class CompositeType
{
bool boolValue = true;
string stringValue = "Hello "; [DataMember]
public bool BoolValue
{
get { return boolValue; }
set { boolValue = value; }
} [DataMember]
public string StringValue
{
get { return stringValue; }
set { stringValue = value; }
}
} #region 新增stu类 [DataContract]
public class Student1
{
string name; [DataMember]
public string Name
{
get { return name; }
set { name = value; }
}
} public class Student2
{
string name = "张三"; public string Name
{
get { return name; }
set { name = value; }
} } #endregion
}

上面这段代码要注意以下几方面:

2-1、服务契约

接口IService1前面加了 [ServiceContract] ,意思是把这个接口(包括继承这个接口的类)声明为服务契约,服务契约是对客户端而言的,就是这个接口 暴露 在客户端面前,就是让客户端可得见这个接口。但看得见接口不表达可以看得见接口里声明的方法,这是两回事(原因很简单,就算接口是可见的,但里面的方法也有一些是可见另一些不可见的嘛),如果想把方法也声明为对客户端可见的,得在声明方法的签名加 [OperationContract],这也叫服务契约。总结:服务契约有两种,[ServiceContract]是声明接口、类对客户端可见的,[OperationContract]是具体声明类里的那些方法对客户端可见。

2-2、数据契约

     接口自带的类CompositeType 和我们自定义的类Student1、Student2 前面都加了 [DataContract] ,意思是把这个类声明为数据契约,这样子客户端就可以用这个类去定义变量了。这里说的类和上面说的类所指的具体内容稍稍有点不同,上面的类偏向于可以调用类中的方法,这里的类偏向于指作为一种类型(就像int string 等类型),可以用来定义变量。类对客户端可见,但类中的字段、变量则不一定(原理和上面说的一样),要在想暴露在客户端的字段、变量前面加 [DataMember] 。但不能在方法(如构造函数)前面加[DataMember],因为函数只能声明为服务契约嘛(加[OperationContract]),而服务契约只能在 [ServiceContract] 下面声明。总结:数据契约有两种,[DataContract]是声明类或结构的,[DataMember] 是声明类或结构中具体的字段或属性(推荐用属性)对客户可见。

再来看Service.svc.cs文件。可以看到里面只是定义了一个继承 IService1接口的类 Service1,主要内容就是实现IService1接口里声明的方法。代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Serialization;
using System.ServiceModel;
using System.ServiceModel.Web;
using System.Text; namespace WcfService_1
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码、svc 和配置文件中的类名“Service1”。
// 注意: 为了启动 WCF 测试客户端以测试此服务,请在解决方案资源管理器中选择 Service1.svc 或 Service1.svc.cs,然后开始调试。
public class Service1 : IService1
{
public string GetData(int value)
{
return string.Format("You entered: {0}", value);
} public CompositeType GetDataUsingDataContract(CompositeType composite)
{
if (composite == null)
{
throw new ArgumentNullException("composite");
}
if (composite.BoolValue)
{
composite.StringValue += "Suffix";
}
return composite;
} #region 新增方法 public Student1 StudentWriteName(string name)
{
Student1 s1 = new Student1();
s1.Name = name; return s1;
} #endregion
}
}

里面的定义的类或者是实现的方法前面都没有加什么 [ServiceContract][OperationContract] 声明,因为类继承的接口已经声明了,所以类就不用再声明[ServiceContract] 了,实现接口的方法 也不用加[OperationContract]

可能大家有一个问题,那就是在这个类里添加自己的方法行不?答案是可以,但没意义。因为在这个类作为服务端(可以这么认为),里面的方法就是为了给客户端使用的,那么就要在声明的方法前加[OperationContract],但只有加了 [ServiceContract]属性的类里的方法才能那样子,又因为类是继承了一个具有 [ServiceContract]属性的接口了的,所以这个类就不用再加 [ServiceContract]属性了(已经有了),一句话,就是在 继承了 实现了服务契约的接口 的类里没必要。添加自己的方法。要加就先加在接口里。

当然,如果你不用接口(可以不用接口的,用接口只是为了更面向对象而已),直接在类里定义服务契约的话,直接把接口里的那些[ServiceContract] 、 [OperationContract] 照搬过来就行了。

第三步:分析ConsoleApp项目

一句话总结:红框框里的那个东西叫服务引用,也就是客户端,也就是靠它才能调用服务端提供的方法和自定义类型。添加方法如下:

右键点击 ConsoleApp项目,选择 添加服务引用(使用默认名称),会弹出一方框:

点右边的 “发现”,就会检索出发现的服务并显示在左边,并把地址也显示出来。点左边 服务里发现的服务左边的三角形,一级级打开,赫然发现里面就是 Service1 类,再往下就是 Service1类继承的接口 IService1 !对应右边就是该服务提供给客户端调用的方法!

打开ConsoleApp里的Program.cs文件,里面代码很简单,定义一个student1类的变量,然后获取她的一个属性值并输出。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace ConsoleApp_1
{
class Program
{
static void Main(string[] args)
{
//定义通信管道 client,就是通过它来调用服务端提供的方法
ServiceReference1.Service1Client client = new ServiceReference1.Service1Client();
string str = client.GetData(1);
Console.WriteLine(str); //服务端提供给客户端使用的类
ServiceReference1.Student1 stu1 = new ServiceReference1.Student1();
stu1 = client.StudentWriteName("张三");
Console.WriteLine("我的姓名:"+stu1.Name); Console.ReadKey(); }
}
}

ServiceReference1(其实是一个命名空间)就是刚才添加的服务引用的名称。当输入ServiceReference1.的时候,后面就会出现智能感知到的这个命名空间能引用的类型、接口等内容(有时候没有出现如你所愿的类型时,右键点WcfService1项目选择重新生成,然后右键点ConsoleApp项目的ServiceReference1选择更新服务引用,这个很容易忘掉,每对服务端做了什么修改都要这么来一下)。图如下:

里面还定义了一个什么通信管道的东西,这个东西估计就是把服务端提供的可工客户端引用的方法封装成了一个类,通过这个类就可以调用那些方法了,详情见代码。

最后按F5运行,没什么问题吧?没有输出?这是正常的,如果你想看到输出结果,可以先对Str属性赋值(stu.Str = "123";)。嘿嘿,那就算完成了。

转载:http://www.cnblogs.com/zouzf/archive/2012/02/28/2370667.html

最后还有两个问题:

1、在IService1接口里自定义了两个类 Student1 和Student2,为什么输入ServiceReference1.后的智能感知里只有 student1类型而没有student2类型?因为在IService1接口里有一个方法 public Student1 StudentWriteName(string name);,这个方法的返回值是 studnet1类型,这就是差别。。。。就是你想在客户使用在服务端里自定义的类型,除了用[DataContract]声明之外,还要有一个服务契约(方法)的返回值是这个类型的。至于为什么,我也不知道。。求答案

2、student1类里的Str属性有一个Get{return str;}方法,而字段str是有默认值的,为什么在Program.cs里定义了一个student1类的变量后,获取Str属性的值并输出时是空白?也就是说属性的默认值发生了丢失!为什么?不知道,求答案。。。

 

C#之WCF入门1—简单的wcf例子的更多相关文章

  1. 1.WCF学习--创建简单的WCF服务

    一.基本了解WCF 1.面向服务代表的是一种设计理念,和面向对象.面向组件一样,体现的是一种对关注点进行分解的思想,面向服务是和技术无关的 2.WCF需要依存一个运行着的宿主进程,服务寄宿就是为服务指 ...

  2. WCF入门(十一)---WCF安全

    一个强大的WCF服务安全系统,拥有两种安全模式或级别预期的客户端可以访问的服务.这是常见的分布式事务的安全威胁正在放缓,在很大程度上由WCF决定. 关键的安全功能 WCF服务有四个主要的安全功能,如下 ...

  3. WCF入门(十)---WCF事务

    事务处理在WCF(Windows Communication Foundation)是一套遵循一些性质,统称为ACID的操作.这里,如果一个操作出现故障,整个系统就会自动失败.如网上订单生成,就可能使 ...

  4. WCF入门(九)--WCF实例管理

    这组由Windows通讯基础(WCF)结合一组消息(客户端请求)服务实例所采用的技术被称为实例管理.一个完全由三种类型实例激活支持WCF,它们如下所述. 1.每个调用服务 每次调用服务是Windows ...

  5. WCF入门(八)---WCF服务绑定

    WCF服务绑定是一个集合,每个元素定义了服务与客户端进行通信方式的几个元素.传输元素和一个消息编码元素各自结合两个最重要的组成部分.这里是WCF服务绑定常用的列表. 基础绑定 基础约束是由basicH ...

  6. WCF入门(四)---WCF架构

    WCF是一个分层架构,为开发各种分布式应用的充分支持.该体系结构在下面将详细说明. 约定 约定层旁边就是应用层,并含有类似于现实世界的约定,指定服务和什么样的信息可以访问它会使操作的信息.约定基本都是 ...

  7. WCF入门(三)---WCF与Web服务/Web Service

    下面列出了WCF和Web服务之间存在一些重大差异. 属性:WCF服务是通过定义ServiceContract和OperationContract属性,而在Web服务,WebService和WebMet ...

  8. java入门---简介&简单输出小例子&开发前准备

        Java是由Sun Microsystems公司于1995年5月推出的Java面向对象程序设计语言和Java平台的总称.由James Gosling和同事们共同研发,并在1995年正式推出.J ...

  9. 无废话WCF入门教程六[一个简单的Demo]

    一.前言 前面的几个章节介绍了很多理论基础,如:什么是WCF.WCF中的A.B.C.WCF的传输模式.本文从零开始和大家一起写一个小的WCF应用程序Demo. 大多框架的学习都是从增.删.改.查开始来 ...

随机推荐

  1. day06-codes and exercise in class

    # Author: Ghost # Email: jiaci.liu@gmail.com ''' 1-Review of last week 2-interface class, abstract c ...

  2. (转)service apache2 restart失败

    https://askubuntu.com/questions/431925/how-to-restart-apache2-when-i-get-a-pid-conflict sudo kill -9 ...

  3. [原创]python高可用程序设计方法

    有时候程序上的bug会导致程序引发诸如段错误的情况而导致程序异常退出,这时用crond服务来检测,就会有一段时间程序处于不可用的情况,为了增强程序的可用性,我们可以让子进程处理业务,而让主进程检测子进 ...

  4. [数据结构]_[C/C++]_[链表的最佳创建方式]

    场景 1.链表在C/C++里使用非常频繁, 因为它非常使用, 可作为天然的可变数组. push到末尾时对前面的链表项不影响. 反观C数组和std::vector, 一个是静态大小, 一个是增加多了会对 ...

  5. GDAL2.1.1库在Ubuntu14.04下编译时遇到的问题处理方法

    不用作任何调整,直接在Linux下编译GDAL2.1.1源码的步骤是: $ ./configure $ make $ make install 非常简单,这样也能正常生成gdal动态库.静态库,如果想 ...

  6. java 类装饰

    package TestIo; public class Test8 { public static void main(String[] args) { System.out.println(&qu ...

  7. ubuntu14.04上安装Java

    apt-get安装 sudo add-apt-repository ppa:webupd8team/java sudo apt-get update sudo apt-get install orac ...

  8. 用Scanner读文本文件内容

    import java.io.File; import java.util.Scanner; class Demo { public static void main(String[] args) t ...

  9. nested class 例子

    #include<iostream> using namespace std; /* start of Enclosing class declaration */ class Enclo ...

  10. 180706-BigDecimal除法的精度问题

    BigDecimal除法的精度问题 在使用BigDecimal的除法时,遇到一个鬼畜的问题,本以为的精度计算,结果使用返回0,当然最终发现还是自己的使用姿势不对导致的,因此记录一下,避免后面重蹈覆辙 ...