WCF4.0新特性体验(11):服务发现WS-Discovery之设定FindCriteria-阿里云开发者社区

开发者社区> 开发与运维> 正文
登录阅读全文

WCF4.0新特性体验(11):服务发现WS-Discovery之设定FindCriteria

简介:
    继WCF4.0新特性体验(10):服务发现WS-Discovery之简单的Ad hoc Service Discovery 之后,今天我们将学习WCF4.0服务发现WS-Discovery的第二部分:WCF4.0新特性体验(11):服务发现WS-Discovery之设定FindCriteria。
     本节里我会介系统介绍服务发现提供了设置客户端查找条件的机制,也就是DiscoveryClient如何在搜索服务的时候使用特定的条件来查找特定的服务。开始部分我会补充介绍一下WCF动态服务发现的终结点内容。然后是设定查找范围(Scope)FindCriteria的概念,以及实现过程。
【1】Find 与FindCriteria:
   Find是客户端查找一个或多个服务的重要动作。客户端在查找服务的过程中,需要发送一个Probe探测消息,匹配服务的消息会回发一个ProbeMatch匹配应答消息给客户端。
【2】DiscoveryClient:
 DiscoveryClient类定义了一些方法,它实现了动态查询服务的机制。利用这个类型,我们可以很方便地在客户端实现服务的查找工作。它定义了一个Find方法,这个方法包含一个异步的实现FindAsync 。两个方法都接受一个FindCriteria 参数,然后返回一个FindResponse 的查询结果。如果查询到匹配的服务,这里会包含一些必要的服务信息,比如地址Address。
【3】服务发现终结点介绍:
  在WCF4.0新特性体验(3):标准终结点(Standard Endpoints)一节里我们已经介绍过了WCF4.0提供的服务终结点。WCF4.0提供了8个已经定义好的标准终结点,其中与WS-Discovery服务动态发现相关终结点包含4个:announcementEndpoint、 discoveryEndpoint、udpAnnouncementEndpoint、udpDiscoveryEndpoint。各个终结点的作用如下表:
名称 描述
mexEndpoint 预定义了一个元数据交换节点,默认使用IMetadataExchange 契约和mexHttpBinding 绑定,address为空。
announcementEndpoint 为discovery的声明功能预定义的标准终结点,在使用此终结点用户需要额外配置address和binding。
discoveryEndpoint 为discovery 操作,预定义的一个标准终结点,在使用此终结点用户需要额外配置address和binding。
udpAnnouncementEndpoint 为使用UDP binding和WS-Discovery 协议规定的多播地址的discovery的声明功能提供的预定义的终结点。它继承自announcementEndpoint。
udpDiscoveryEndpoint 为discovery 操作提供的预定义的终结点。比如通过UDP binding和WS-Discovery 协议规定的多播地址的发现(find) 和解析(resolve。它继承自DiscoveryEndpoint。
workflowControlEndpoint 定义了一个控制workflow 实例执行状态的标准终结点(create、run、suspend、terminate等等)。
webHttpEndpoint 定义了一个配置了WebHttpBinding 和WebHttpBehavior的标准终结点。它继承自WebServiceEndpoint。在编写REST service使用。
webScriptEndpoint 定义了一个配置了WebHttpBinding 和WebScriptEnablingBehavior的标准终结点。它用于编写ASP.NET AJAX程序。
【4】FindCriteria:
FindCriteria 定义了几个属性,他们可以设置查询条件,我们可以来指定查询条件。此外这个还可以指定查询条件的持续时间。如果我们要查找符合其中一些条件而不是全部条件的服务,就必须自定义Find逻辑,或者使用多个查询来实现。
查询条件包含:
  • ContractTypeNames :选择性的,目标服务使用的契约名称。如果只有一个契约符合条件,则匹配服务的所有契约都会应答。因为一个WCF服务可以定义多个终结点,每个终结点只能包含一个契约Contract。
  • Scopes :选择性的,Scopes是服务终结点使用的绝对的URI。这个条件在查询多个终结点使用了同一个契约的时候,非常有用。我们可以查询匹配特定URI的终结点。如果我们指定多个Scope,则只有满足所有Scope的终结点才会响应查询。
  • ScopeMatchBy:在Probe探测消息匹配Scope的时候指定特定的匹配规则,支持一下五种规则:
    • ScopeMatchByExact:会做精确的大小写匹配。
    • ScopeMatchByPrefix:匹配前缀,只要URI前缀部分相同即可匹配。
    • ScopeMatchByLdap:使用LDAP URL来匹配Scope。
    • ScopeMatchByUuid:使用UUID 字符串来匹配Scope。
    • ScopeMatchByNone:匹配服务的时候不需要Scope.
    默认情况下,如果不指定ScopeMatchBy,就会使用ScopeMatchByPrefix规则。
结束查询包含:
  1. Duration:等待服务响应的最长等待时间。默认是20秒。
  2. MaxResults:等待响应消息的最大数目,如果在Duration 持续时间内返回了期望的消息数目,则查询操作终止
【5】FindResponse:
 FindResponse包含一个EndPoint集合属性,这个属性里包含了所有回发应答ProbeMatch探测消息的服务地址。如果没有匹配的地址,集合就会为空。如果有匹配的服务,那么每个应答消息都会存储在一个EndpointDiscoveryMetadata对象里,这个对象里包含了服务的Address、Contract以及其它的一些信息。
【6】示例代码分析:
  第一节代码里,我们给出了简单的基于Ad-Hoc模式的服务发现例子,使用的是基于Contract作为查询条件。另外我们也可以增加其它的限制条件,必须Scope来提高匹配精确度。
  首先,这里如果使用Scope,服务在发布以前就必须把每个终结点EndPoint和特定的Scope关联。WCF4.0里,我们可以在<endpointDiscovery> behavior 里进行设置,当然也可以通过代码实现。下面我们就来详细讲解一下详细的实现过程。
【6.1】服务端:
  我们这里服务端端对于其中的一个终结点,配置了Scope,其余终结点没有指定。具体配置信息如下:
<system.serviceModel> 
        <services> 
            <service name="WCFService.WCFService"> 
                <endpoint address="http://localhost:8000/ws"        binding="wsHttpBinding" contract="WCFService.IWCFService" behaviorConfiguration="epBehavior" /> 
                <endpoint address="http://localhost:8000/basic" binding="basicHttpBinding" contract="WCFService.IWCFService" /> 
                <endpoint address="net.tcp://localhost:8001/tcp" binding="netTcpBinding" contract="WCFService.IWCFService" /> 
                <!-- add a standard UDP discovery endpoint--> 
                <endpoint name="udpDiscovery" kind="udpDiscoveryEndpoint"/> 
            </service> 
        </services> 
        <behaviors> 
            <serviceBehaviors> 
                <behavior> 
                    <serviceDiscovery/> 
                    <!-- enable service discovery behavior --> 
                </behavior> 
            </serviceBehaviors> 
            <endpointBehaviors> 
                <behavior name="epBehavior"> 
                    <endpointDiscovery> 
                        <!-- scopes associated with this endpoint behavior --> 
                        <scopes> 
                            <add scope="http://localhost:8000/"/> 
                        </scopes> 
                    </endpointDiscovery> 
                </behavior> 
            </endpointBehaviors> 
        </behaviors> 
    </system.serviceModel>
【6.2】客户端:
  客户端可以在运行时,动态查询匹配的终结点。我们可以增加Scope条件到查询FindCriteria的属性里。下面例子展示了如何使用ScopeMatchByPrefix来查询服务终结点。我们先动态查询所有可用的终结点信息,然后打印出来,最后使用Scope匹配一个特定的终结点:代码如下:
// 创建DiscoveryClient 
                        DiscoveryClient discoveryClient = new DiscoveryClient("udpDiscoveryEndpoint"); 
                        // Find ICalculatorService endpoints in the specified scope 
                        // 特定范围内查找IWCFService终结点 
                        FindCriteria findCriteria = new FindCriteria(); 
                        //findCriteria.Scopes.Add(scope); 
                        FindResponse findResponse = discoveryClient.Find(findCriteria); 
                        //打印所有终结点信息 
                        Console.WriteLine("All Endpoints:"); 
                        Console.ForegroundColor = ConsoleColor.Yellow; 
                        foreach (EndpointDiscoveryMetadata edm in findResponse.Endpoints) 
                        { 
                                Console.WriteLine("[Address]: {0},[Contract]: {1}", 
                                edm.Address, edm.ContractTypeNames[0].Name); 
                        } 
                        //定义Scope 
                        Uri scope = new Uri("http://localhost:8000/"); 
                        findCriteria.Scopes.Add(scope); 
                        findResponse = discoveryClient.Find(findCriteria); 
                        //打印所有终结点信息 
                        Console.WriteLine("Special Endpoints:"); 
                        Console.ForegroundColor = ConsoleColor.Red; 
                        foreach (EndpointDiscoveryMetadata edm in findResponse.Endpoints) 
                        { 
                                Console.WriteLine("[Address]: {0},[Contract]: {1}", 
                                edm.Address, edm.ContractTypeNames[0].Name); 
                        } 
【6.3】运行结果:
     启动Host,运行客户端结果显示如下:
【7】总结:
  使用Scope,我们可以更容易地查找特定的服务终结点EndPoint。尤其是当一个服务实现多个终结点的时候。另外服务发现也允许扩展,我们可以给服务在一个EndPoint上添加特定的元数据。这些MetaData信息业会在应答消息里返回给客户端。
     下面给出本文的例子代码,供大家参考:
参考文章:


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


版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

分享: