最近项目中接触了一点WCF的知识,也就是怎么调用WCF服务,上网查了一些资料,很快就搞出来,可是不符合头的要求,主要有以下几个方面:
①WCF的地址会变动,地址虽变,但是里面的逻辑不变!
②不要引用WCF服务的接口DLL文件,这样会导致一定的耦合性(虽然接口一旦指定就不会改动了)!
1.手动配置服务(这种方式比较傻瓜)
这种方式比较容易,也不多说了,直接给个地址就好了:学习 WCF (6)--学习调用WCF服务的各种方法
2.引用接口DLL,通过地址动态调用WCF服务
1 private void ReleasePC(string clientName, string ttgServiceURL)
2 {
3 try
4 {
5 object[] obj = new object[2] { clientName, clientName };
6 string str = (string)ExecuteMethod<IGatewayWebService>(ttgServiceURL, "ReleasePC", obj);
7 BasicHttpBinding binding = new BasicHttpBinding(BasicHttpSecurityMode.None) { MaxReceivedMessageSize = 6666 };
8 EndpointAddress address = new EndpointAddress(ttgServiceURL);
9 ChannelFactory<IGatewayWebService> ttgService = new ChannelFactory<IGatewayWebService>(binding, address);
10 string result = ttgService.CreateChannel().ReleasePC(clientName, clientName);
11 GatewayReturnDef returnDef = JSONSerializer.FromJSON<GatewayReturnDef>(result);
12 if (returnDef.Result)
13 LoggingManager.WriteLog(LogLevel.Information, new LogSeparator("\r\n"), "Release pc success.");
14 else
15 LoggingManager.WriteLog(LogLevel.Error, new LogSeparator("\r\n"), "Release pc failed.");
16 }
17 catch (Exception exception)
18 {
19 LoggingManager.WriteLog(LogLevel.Exception, new LogSeparator("\r\n"), "Error when invoke ReleasePC throw exception.", exception.Message, exception.InnerException);
20 }
21 }
Note:这种方式需要引用WCF服务的提供的元数据,也就是接口DLL的引用,如果我在接口添加里面添加一个方法,那么我所引用的接口DLL也必须发生改变,这就导致了两者的耦合性升高,这种情况是不允许发生的,但也没说不允许使用,因为大部分情况下接口一旦定下来就不会变动,所以不要担心接口DLL的改动!
Note:你也可以把这样的方法写成泛型,那么他的可扩展性就变得更好了!
3.通过地址获取元数据,动态生成代理类,调用WCF服务
1 private void ReleasePC(string ttgServiceURL, object[] args)
2 {
3 try
4 {
5 string serviceURL = ttgServiceURL + "/?wsdl";
6 DynamicProxyFactory factory = new DynamicProxyFactory(serviceURL);
7 DynamicProxy dynamicProxy = factory.CreateProxy("IGatewayWebService");
8 string result = (string)dynamicProxy.CallMethod("ReleasePC", args);
9 GatewayReturnDef returnDef = JSONSerializer.FromJSON<GatewayReturnDef>(result);
10 dynamicProxy.Close();
11 if (returnDef.Result)
12 LoggingManager.WriteLog(LogLevel.Information, new LogSeparator("\r\n"), "Release pc success.");
13 else
14 LoggingManager.WriteLog(LogLevel.Error, new LogSeparator("\r\n"), "Release pc failed.");
15 }
16 catch (Exception exception)
17 {
18 LoggingManager.WriteLog(LogLevel.Exception, new LogSeparator("\r\n"), "Error when invoke ReleasePC throw exception.", exception.Message, exception.InnerException);
19 }
20 }
Note:如果上面两种方法不能适应项目需要的话,那么这就是最后的武器了,不过它的性能有点差,因为首先要获取元数据,然后再自动生成代理类,所以性能会不好!
这种方式是通过服务地址获取WCF的元数据(wsdl),然后在再客户端动态生成代理类(第一种情况已经帮我们生成好了代理类,第二种情况有了元数据,就差代理类,第三种是既没元数据又没代理类),所以关键的地方就是获取WCF服务的元数据!
通过前辈们提供的公用类库,我们很容易的得到WCF服务提高的元数据,下面就是生成动态生成代理类的项目文件:动态创建WCF服务代理类
注意事项(如果采用第三种方式获取元数据,需要注意下面的问题):
如果你是个分布式服务的话,服务端的URL地址就不能使用“localhost”地址了,因为它是一个回传地址“127.0.0.1”,所以必须把它改为局域网地址或者公网地址,这个很重要呀,一定要切忌...
学习WCF刻不容缓啦,o(∩_∩)o 哈哈...