我的WCF摸爬滚打之路(1)

时间:2021-07-10 10:05:19

等了好久终于等到今天!盼了好久终于把梦实现……哈哈,仅以此歌词来庆祝我为期3天的wcf学习之路圆满结束。

今天写这个文章的目的在于记录一下我自己在学习WCF的时候碰到的一些问题,俗话说,好记心不如烂笔头嘛。也为看见我这篇文章的有缘人(正在wcf中探索的人们,提供一点借鉴吧)。

还是老样子,图文并茂的文章才是好文章(我自己定义的,勿喷!)。那么接下来我将把我这一路的细节以及碰到的问题和解决方法一一展示出来。

我在学习wcf的时候第一件事就是自己做了个wcf的例子,因为wcf的传输协议有很多,比如说http、tcp。我主要讲讲tcp协议的wcf吧,因为这个确实比较麻烦的一个东西,有时候出现bug也是一些稀奇古怪的exception。废话不多说直接上图。

新建一个项目

我的WCF摸爬滚打之路(1)

新建好了的wcf服务库如下图

我的WCF摸爬滚打之路(1)

会自动生成几个类文件。这几个类文件分别是两个interface和两个interface的实现类(需要自己手动实现)。可能有的人会问,我怎么只生成了一个interface和它的实现类额?因为我想实现的是wcf通过一个servicehost和一个端口同时发布多个服务出来。另外两个是我自己新建的 。哈哈……
看看具体的代码都是啥!很简单只是声明了一个GetData(int value)函数。翻翻书或者百度一下都知道wcf的基本概念:以契约(Contract)来定义双方通信,契约必须以接口的形式体现。这里面比较重要的两个特性[ServiceContract]、[OperationContract]。一个服务契约,一个操作契约,只有定义了这两个别人才认识你是个wcf服务撒。用这个时必须引用命名空间:using System.ServiceModel;
 namespace WcfServiceLibrary
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的接口名“IService1”。
[ServiceContract]
public interface IService1
{
[OperationContract]
string GetData(int value); } }

既然wcf叫服务那他总的有服务吧!不然怎么服务于我。别急,上代码:上面是契约、下面是服务。实际的服务代码要由契约接口派生实现。

 namespace WcfServiceLibrary
{
// 注意: 使用“重构”菜单上的“重命名”命令,可以同时更改代码和配置文件中的类名“Service1”。
public class Service1 : IService1
{
public string GetData(int value)
{
if (value==)
{
throw new NotImplementedException();
}
return string.Format("You entered: {0}", value);
} }
}

以上工作完成后,wcf的创建算是完工了。接下来的重点,是建立wcf的宿主程序。
我就做了和控制台程序作为wcf说的宿主程序!

一般常见的是通过config文件进行配置,以为要深究的话,config文件是在有很东西要讲,而我也只是初出茅庐。所以我就一代码的形式进行配置文件的生成。而且只需要几个常用的属性就ok。且看代码

1.我先实例化绑定形式NetTcpBinding

  private static NetTcpBinding netTcpBinding = new NetTcpBinding();
public static NetTcpBinding InitNetTcpBinding()
{
netTcpBinding.Security.Mode = SecurityMode.None;
netTcpBinding.ReceiveTimeout = TimeSpan.Parse("00:10:00");
netTcpBinding.MaxBufferPoolSize = ; //
netTcpBinding.MaxBufferSize = ;
//netTcpBinding.MaxConnections = 10;
//netTcpBinding.PortSharingEnabled = true; netTcpBinding.ReaderQuotas.MaxDepth = ;
netTcpBinding.ReaderQuotas.MaxStringContentLength = ;
netTcpBinding.ReaderQuotas.MaxArrayLength = ;
netTcpBinding.ReaderQuotas.MaxBytesPerRead = ;
netTcpBinding.ReaderQuotas.MaxNameTableCharCount = ;
netTcpBinding.MaxReceivedMessageSize = ;
return netTcpBinding;
}

2.我通过反射的形式拿到wcf的契约和服务。

  public static void LoadAssemBly(string assemblyName)
{
Assembly assem = Assembly.Load(assemblyName);
Dictionary<Type, Type> svTypes = new Dictionary<Type, Type>();
List<TypeInfo> list = assem.DefinedTypes.ToList();
foreach (TypeInfo typeInfo in list)
{
if (typeInfo.Attributes.ToString().IndexOf("Abstract") >= )
{
TypeInfo tempK = typeInfo;
var result = (from t in list where t.Name==(tempK.Name.Substring()) select t).ToList();
TypeInfo tempV = result[];
svTypes.Add(tempK, tempV);
} }
string serviceAddress = string.Format("net.tcp://{0}:{1}", "localhost", "");
string endpointAddress = string.Empty;
string tName = string.Empty;
foreach (var item in svTypes)
{
tName = item.Key.Name.Substring();
endpointAddress = serviceAddress +"/"+ tName; ServiceHost serviceHost = new ServiceHost(item.Value, new Uri(endpointAddress)); //加载元数据节点
ServiceMetadataBehavior smb = new ServiceMetadataBehavior();
serviceHost.Description.Behaviors.Add(smb);
serviceHost.AddServiceEndpoint(typeof(IMetadataExchange), MetadataExchangeBindings.CreateMexTcpBinding(), "mex"); serviceHost.AddServiceEndpoint(item.Key, netTcpBinding, endpointAddress); serviceHost.Opened += delegate
{
msg.AppendLine(string.Format("{0}开始监听 Uri 为 :{1}/mex", tName, endpointAddress.ToString()));
};
serviceHost.Open();
serviceHosts.Add(serviceHost); }

这样我们得宿主程序就完成了。运行一下看看结果
我的WCF摸爬滚打之路(1)

这样看来是没问题了,那么们测试一下添加服务引用是否能找到服务。如果找到了就说明确实是没问题了。

我的WCF摸爬滚打之路(1)

OK,搞定!看似一帆风顺。其实这中间我遇到了很多问题,下次再专门写个博客讲解吧

《转载注明出处!》