WCF分布式开发步步为赢(5)服务契约与操作重载

简介:
  继上一节WCF分布式开发步步为赢系列的(4):WCF服务可靠性传输配置与编程开发,本节我们继续学习WCF分布式开发步步为赢的第(5)节:服务契约与操作重载。这里我们首先讲解OOP面向对象的编程中方法重载,重载的意义,WCF服务编程开发如何实现操作重载,随后是代码分析部分,给出了服务端服务契约定义和实现操作重载的注意的问题和实现过程,然后详细介绍了客户端实现操作重载的方式。最后是本文的总结部分。本节的结构是:【1】 重载概念【2】操作 重载【3】代码实现分析【4】运行结果【5】总结
【1】 重载概念:
【1.1】什么是重载(OverLoad):
    所谓重载是指同一个方法名可以对应着多个方法的实现。这些方法的名字相同,但是方法的参数的类型不同。这就是方法重载的概念。函数方法类和对象的应用尤其重要。
 方法重载要求编译器能够唯一地确定调用一个方法时应执行哪个方法代码,即采用哪个方法实现。确定方法实现时,要求从方法参数的个数和类型上来区分。这就是说,进行方法重载时,要求同名方法在参数个数上不同,或者参数类型上不同。否则,将无法实现重载。
    关于重载一定要注意:重载方法的参数类型和参数个数一定要不同(即:要么参数的类型不同,要么参数的个数不同,要么参数的类型和个数都不同),否则,编译器就不知道该调用那个方法了。
   方法重载的好处就是相同的方法,带来不同的结果和实现,这里我们可以根据传递参数的不同来决定调用飞方法。这是编译时多态的一种实现机制。
【1.2】C#类方法重载示例:
    我们这里给出一个简单的c#语言实现的方法重载的列子,这里对于 SayHelloOverLoading方法,同一个类里给出的三个方法的参数个数不同。内部实现也不同。具体代码如下:
// 3.面向对象里的类,如何实现操作重载,和WCF服务类里的操作重载做对比
     public   class  ClassOverLoading
    {
        
public  ClassOverLoading()
        { 

        }
        
// 掩饰方法重载,分别实现三个方法,C#等面向对象的语言提供了方法重载机制的支持。
         public   string  SayHelloOverLoading()
        { 
            
// 编写代码
             return   " Hello,This an C# class overloading demo " ;
        }
        
// 类里的方法重载不需要别名
         public   string  SayHelloOverLoading( string  name)
        {
            
// 编写代码
             return   " Hello: "   +  name  +   " This an C# class overloading demo " ;
        }

        
public   string  SayHelloOverLoading( string  firstName,  string  lastName)

        {
            
// 编写代码
             return   " Hello: "   +  firstName  +  lastName  +   " This an C# class overloading demo " ;
        }

    }
 
【2】操作 重载:
【2.1】操作重载:
    WCF服务支持核心的Web 服务协议,同样其元数据交换也是基于XML语言描述,客户端通过WSDL文件来了解服务方法相关的信息,包括参数的个数、类型、返回值、调用顺序等重要信息。由于WSDL不支持方法的重载,因此我们的WCF服务操作重载就无法通过WSDL暴露给客户端。如果我们在服务契约里定义了方法的重载,编译可以正常通过,但是启动服务宿主就会抛出System.InvalidOperationException异常,如下图:
  因此我们不能在WCF服务类了定义和实现方法重载,否则无法暴露为服务操作。
【2.2】解决办法:
   WCF给我们提供了一个解决办法,让我们可以在WCF服务类里使用服务操作的重载。WCF定义了一个机制OperationContract,使用OperationContract特性的Name属性,为操作指定别名:
[AttributeUsage(AttributeTargets.Method)]
public   sealed   class  OperationContractAttribute : Attribute
{
   
public   string  Name
   {
get ; set ;}

   
// 更多成员

}

【3】代码实现分析:
    下面我们来给出一个具体的WCF服务实现操作重载,包括服务定义、宿主配置、客户端引用和测试的完整过程。
【3.1】服务契约:
    定义了服务契约 IWCFOverLoadingService,分别给出SayHelloOverLoading操作契约的3种不同定义和WCFService服务类里的实现。具体代码如下:
     // 1.服务契约,操作契约重载
    [ServiceContract(Namespace  =   " http://www.cnblogs.com/frank_xl/ " )]
    
public   interface  IWCFOverLoadingService
    {
        
// 操作契约
        [OperationContract(Name  =   " SayHelloOverLoading1 " )]
        
string  SayHelloOverLoading();
        
// 操作契约
        [OperationContract(Name  =   " SayHelloOverLoading2 " )]
        
string  SayHelloOverLoading( string  name);
        
// 操作契约
        [OperationContract(Name  =   " SayHelloOverLoading3 " )]
        
string  SayHelloOverLoading( string  firstName,  string  lastName);

    }
    
// 2.服务类,集成接口。实现契约
     public   class  WCFService : IWCFOverLoadingService
    {
        
// 实现接口定义的方法
         public   string  SayHelloOverLoading()
        {
            Console.WriteLine(
" Hello! ,This an overloading demo for WCF Service  " );
            
return   " Hello! This an overloading demo for WCF Service   " ;
        }
        
// 实现接口定义的方法
         public   string  SayHelloOverLoading( string  name)
        {
            Console.WriteLine(
" Hello! {0},This an overloading demo WCF Service  " , name);
            
return   " Hello!  "   +  name  +   " , This an overloading demo for WCF Service  " ;
        }
        
// 实现接口定义的方法
         public   string  SayHelloOverLoading( string  firstName,  string  lastName)
        {
            Console.WriteLine(
" Hello! {0}    {1},This an overloading demo WCF Service " , firstName, lastName);
            
return   " Hello!  "   +  firstName  +   "   "   +  lastName  +   " , This an overloading demo for WCF Service  " ; ;
        }
    }
【3.2】托管宿主:
   自定义托管宿主使用配置文件来定义服务的终结点和元数据交换节点,服务的交换行为等其他属性也在配置文件里给出,我们配置了三种不同的数据服务通信方式,分别是http、tcp、IPC.具体配置信息如下:
<services> 
            <service behaviorConfiguration="WCFService.WCFServiceBehavior" name="WCFService.WCFService"> 
                <endpoint 
                    address="http://localhost:9001/WCFService" 
                    binding="wsHttpBinding" 
                    contract="WCFService.IWCFOverLoadingService"> 
                </endpoint> 
                <endpoint 
                    address="net.tcp://localhost:9002/WCFService" 
                    binding="netTcpBinding" 
                    contract="WCFService.IWCFOverLoadingService"> 
                </endpoint> 
                <endpoint 
                address="net.pipe://localhost/WCFService" 
                binding="netNamedPipeBinding" 
                contract="WCFService.IWCFOverLoadingService"> 
                </endpoint> 
                <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" /> 
                <endpoint address="mex" binding="mexTcpBinding" contract="IMetadataExchange" /> 
                <endpoint address="mex" binding="mexNamedPipeBinding" contract="IMetadataExchange" /> 
                <host> 
                    <baseAddresses> 
                        <add baseAddress="http://localhost:9001/"/> 
                        <add baseAddress="net.tcp://localhost:9002/"/> 
                        <add baseAddress="net.pipe://localhost/"/> 
                    </baseAddresses> 
                </host> 
            </service> 
        </services> 
        <behaviors> 
            <serviceBehaviors> 
                <behavior name="WCFService.WCFServiceBehavior"> 
                    <serviceMetadata httpGetEnabled="true" /> 
                    <serviceDebug includeExceptionDetailInFaults="false" /> 
                </behavior> 
            </serviceBehaviors> 
        </behaviors>
【3.3】客户端服务引用:
    我们来分别添加对服务端的引用,首先启动托管宿主程序。然后使用Visual Studio2008工具直接添加服务引用,你也可以使用svcUtil.exe工具,如图所示:
    客户端输入服务的基地址,查找服务,成功够我们可以看到服务契约的信息,这里显示的3个操作名称已经不同,实际上这里给出的是三个不同名称的服务方法。输入命名空间,确定即可完成。
【3.4】代理代码:
    客户端反序列化生成的服务契约等信息,我们查看操作契约对应的客户端方法名称以及改变,这样一来,客户端就没有实现对应的方法重载,也就不能使用重载带来的优势,也即是编译时多态的特性。手动更改客户端服务代理类和服务契约代码,使之支持操作方法重载,代码如下:
public   interface  IWCFOverLoadingService {
        
        [System.ServiceModel.OperationContractAttribute(Name 
=   " SayHelloOverLoading1 "  ,Action = " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading1 " , ReplyAction = " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading1Respon "   +
            
" se " )]
        
string  SayHelloOverLoading();

        [System.ServiceModel.OperationContractAttribute(Name 
=   " SayHelloOverLoading2 " , Action  =   " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading2 " , ReplyAction  =   " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading2Respon "   +
            
" se " )]
        
string  SayHelloOverLoading( string  name);

        [System.ServiceModel.OperationContractAttribute(Name 
=   " SayHelloOverLoading3 " , Action  =   " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading3 " , ReplyAction  =   " http://www.cnblogs.com/frank_xl/IWCFOverLoadingService/SayHelloOverLoading3Respon "   +
            
" se " )]
        
string  SayHelloOverLoading( string  firstName,  string  lastName);
    }
    
    这样我们客户端方法也支持操作方法的重载特性。
【3.5】客户端测试代码:
    为了测试操作契约,我们在客户端应用里添加了部分的测试代码,这里为了测试服务端定义的不同的操作。我们分组按照协议给出了测试的代码:
 
             // 实例化客户端服务代理Tcp
            ServiceOverLoadingTcp.WCFOverLoadingServiceClient wcfServiceProxyTcp  =
                
new  ServiceOverLoadingTcp.WCFOverLoadingServiceClient( " WSHttpBinding_IWCFOverLoadingService1 " );
            Console.WriteLine(
" Test call service using TCP--------------------. " );
            
// 通过代理调用SayHelloOverLoading服务,分别传递不同的参数,进行测试
            Console.WriteLine(wcfServiceProxyTcp.SayHelloOverLoading());
            Console.WriteLine(wcfServiceProxyTcp.SayHelloOverLoading(
" Frank Xu Lei " ));
            Console.WriteLine(wcfServiceProxyTcp.SayHelloOverLoading(
" Lei " " Xu " ));

            
// 实例化客户端服务代理Http
            ServiceOverLoadingHttp.WCFOverLoadingServiceClient wcfServiceProxyHttp  =
                
new  ServiceOverLoadingHttp.WCFOverLoadingServiceClient( " NetTcpBinding_IWCFOverLoadingService " );
            Console.WriteLine(
" Test call service using Http------------------- " );
            
// 通过代理调用SayHelloOverLoading服务,分别传递不同的参数,进行测试
            Console.WriteLine(wcfServiceProxyHttp.SayHelloOverLoading());
            Console.WriteLine(wcfServiceProxyHttp.SayHelloOverLoading(
" Frank Xu Lei " ));
            Console.WriteLine(wcfServiceProxyHttp.SayHelloOverLoading(
" Lei " " Xu " ));

            
// Debuging
            Console.WriteLine( " Press any key to continue " );
            Console.Read();
【4】运行结果:
    这里分别调用三种服务操作,进行测试。运行的结果如图所示:
【5】总结:
     以上就是本节对WCF服务操作重载的介绍,包括一般重载的基本定义和c#语言中简单的方法重载的实现。然后介绍了WCF操作重载的实现机制、局限性和解决办法,服务契约默认不支持操作方法重载,我们可以利用WCF已有的机制给出方法的别名来解决这个问题。然后给出了包括客户端等完整的测试解决方案,客户端反序列话生成服务类默认不支持服务操作方法重载的,生成的也是服务操作的别名方法。我们在客户端要想使服务代理类支持重载,以利用重载的优势,就需要重新修改客户端服务代理代码。 另外给出本节的实例代码供大家参考:
参考资料:
1.《函数重载》: http://baike.baidu.com/view/534068.htm




 本文转自 frankxulei 51CTO博客,原文链接:http://blog.51cto.com/frankxulei/320409,如需转载请自行联系原作者





相关文章
|
8月前
|
消息中间件 NoSQL Java
Java高级开发:高并发+分布式+高性能+Spring全家桶+性能优化
Java高架构师、分布式架构、高可扩展、高性能、高并发、性能优化、Spring boot、Redis、ActiveMQ、Nginx、Mycat、Netty、Jvm大型分布式项目实战学习架构师之路
|
8月前
|
NoSQL Java Redis
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的分布式锁的功能组件(一)
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的分布式锁的功能组件
101 0
|
23天前
|
数据管理 API 调度
鸿蒙HarmonyOS应用开发 | 探索 HarmonyOS Next-从开发到实战掌握 HarmonyOS Next 的分布式能力
HarmonyOS Next 是华为新一代操作系统,专注于分布式技术的深度应用与生态融合。本文通过技术特点、应用场景及实战案例,全面解析其核心技术架构与开发流程。重点介绍分布式软总线2.0、数据管理、任务调度等升级特性,并提供基于 ArkTS 的原生开发支持。通过开发跨设备协同音乐播放应用,展示分布式能力的实际应用,涵盖项目配置、主界面设计、分布式服务实现及部署调试步骤。此外,深入分析分布式数据同步原理、任务调度优化及常见问题解决方案,帮助开发者掌握 HarmonyOS Next 的核心技术和实战技巧。
186 76
鸿蒙HarmonyOS应用开发 | 探索 HarmonyOS Next-从开发到实战掌握 HarmonyOS Next 的分布式能力
|
3月前
|
NoSQL Java Redis
开发实战:使用Redisson实现分布式延时消息,订单30分钟关闭的另外一种实现!
本文详细介绍了 Redisson 延迟队列(DelayedQueue)的实现原理,包括基本使用、内部数据结构、基本流程、发送和获取延时消息以及初始化延时队列等内容。文章通过代码示例和流程图,逐步解析了延迟消息的发送、接收及处理机制,帮助读者深入了解 Redisson 延迟队列的工作原理。
|
8月前
|
NoSQL Java Redis
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的分布式锁的功能组件(二)
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的分布式锁的功能组件
53 0
|
5月前
|
开发者 云计算 数据库
从桌面跃升至云端的华丽转身:深入解析如何运用WinForms与Azure的强大组合,解锁传统应用向现代化分布式系统演变的秘密,实现性能与安全性的双重飞跃——你不可不知的开发新模式
【8月更文挑战第31天】在数字化转型浪潮中,传统桌面应用面临新挑战。本文探讨如何融合Windows Forms(WinForms)与Microsoft Azure,助力应用向云端转型。通过Azure的虚拟机、容器及无服务器计算,可轻松解决性能瓶颈,满足全球用户需求。文中还提供了连接Azure数据库的示例代码,并介绍了集成Azure Storage和Functions的方法。尽管存在安全性、网络延迟及成本等问题,但合理设计架构可有效应对,帮助开发者构建高效可靠的现代应用。
41 0
|
8月前
|
分布式计算 负载均衡 Java
构建高可用性Java应用:介绍分布式系统设计与开发
构建高可用性Java应用:介绍分布式系统设计与开发
92 0
|
8月前
|
缓存 监控 负载均衡
【分布式技术专题】「缓存解决方案」一文带领你好好认识一下企业级别的缓存技术解决方案的运作原理和开发实战(数据缓存不一致分析)
【分布式技术专题】「缓存解决方案」一文带领你好好认识一下企业级别的缓存技术解决方案的运作原理和开发实战(数据缓存不一致分析)
143 2
|
8月前
|
存储 负载均衡 NoSQL
【分布式技术架构】「Tomcat技术专题」 探索Tomcat集群架构原理和开发分析指南
【分布式技术架构】「Tomcat技术专题」 探索Tomcat集群架构原理和开发分析指南
168 1
|
8月前
|
缓存 应用服务中间件 数据库
【分布式技术专题】「缓存解决方案」一文带领你好好认识一下企业级别的缓存技术解决方案的运作原理和开发实战(多级缓存设计分析)
【分布式技术专题】「缓存解决方案」一文带领你好好认识一下企业级别的缓存技术解决方案的运作原理和开发实战(多级缓存设计分析)
172 1