zoukankan      html  css  js  c++  java
  • Dubbo学习笔记11:Dubbo服务消费方启动流程源码分析

    同理我们看下服务消费端启动流程时序图:

    在《Dubbo整体架构分析》一文中,我们提到服务消费方需要使用ReferenceConfig API来消费服务,具体是调用代码(1)get()方法来生成远程调用代理类。get()方法最终会调用createProxy方法来具体创建代理类,其中createProxy结合时序图的核心代码如下:

    复制代码
    @SuppressWarnings({"unchecked" , "rawtypes" , "deprecation"})
    private T createProxy(Map<String , String> map){
        ...
        if(isJvmRefer){
            ...
        }else{
            ...
            // (1) 当只配置一个服务中心的时候
            if(urls.size() == 1){
                invoker = refprotocol.refer(interfaceClass , urls.get(0));
            }else{
                // (2) 多个服务中心的时候
                ...
            }
        }
        ...
        // (3) 创建代理服务
        return (T)proxyFactory.getProxy(invoker);
    }
    复制代码

    其中refprotocol的定义如下:

    private static final Protocol refprotocol = ExtensionLoader.getExtensionLoader(Protocol.class).getAdaptiveExtension();

    可知refprotocol是Protocol扩展接口的适配器类,这里调用refprotocol.refer(interfaceClass , urls.get(0));实际调用的是Protocol$Adaptive的refer方法。从Protocol$Adaptive的refer方法内部,我们可以发现当前协议类型为registry,也就是这里需要调用RegistryProtocol的refer方法,但是RegistryProtocol被QosProtocolWrapper / ProtocolFilterWrapper / ProtocolListenerWrapper三个wrapper类增强了。所以这里经过一层层调用后,最后调用到了RegistryProtocol的refer方法,其内部主要是调用了doRefer方法,doRefer代码如下:

    复制代码
    private <T> Invoker<T> doRefer(Cluster cluster , Registry registry , Class<T> type , URL url){
        ...
        // (3)
        directory.subscribe(subscribeUrl.addParameter(Constants.CATEGORY_KEY , Constants.PROVIDERS_CATEGORY + "," + Constants.CONFIGURATORS_CATEGORY + "," + Constants.ROUTERS_CATEGORY));
        // (4)
        return cluster.join(directory);
    }
    复制代码

    如上代码(3)的作用是向服务注册中心订阅服务提供者的服务,代码(4)则是调用扩展接口Cluster的适配器类的join方法,根据参数选择配置的集群容错策略。这里我们先讲讲代码(3)的逻辑,看看如何把服务消费方远程服务转换到Invoker,这里结合Zookeeper作为服务治理中心来讲解,首先看看时序图:

    如上时序图步骤(2)(3)(4),从Zookeeper获取服务提供者的地址列表,等Zookeeper返回地址列表后会调用RegistryDirectory的notify方法,代码(6)(7)(8)根据获取的最新的服务提供者url地址转换为具体的invoker列表,也就是每个提供者的url会被转换为一个Invoker对象,具体转换在toInvokers方法中进行:

    复制代码
    private Map<String , Invoker<T>> toInvokers(List<URL> urls){
        Map<String , Invoker<T>> newUrlInvokerMap = new HashMap<String , Invoker<T>>();
        ...
        for(URL providerUrl : urls){
            ...
            Map<String , Invoker<T>> localUrlInvokerMap = this.urlInvokerMap;    //
            Invoker<T> invoker = localUrlInvokerMap == null ? null : localUrlInvokerMap.get(key);
            if(invoker == null){
                try{
                    ...
                    if(enabled){
                        // 这里具体调用dubbo协议转换服务为invoker
                        invoker = new InvokerDelegete<T>(protocol.refer(serviceType , url) , url , providerUrl);
                    }
                }catch(Throwable t){
                    
                }
                if(invoker != null){
                    newUrlInvokerMap.put(key , invoker);
                }
            }else{
                newUrlInvokerMap.put(key , invoker);
            }
        }
        return newUrlInvokerMap;
    }
    复制代码

    如上代码,具体转换服务到invoker对象是通过调用protocol.refer(serviceType , url)来完成的,这里的protocol对象也是Protocol扩展接口的适配器对象,所以调用protocol.refer实际是调用Protocol$Adaptive的refer方法。url中协议默认为dubbo,所以适配器里调用的应该是DubboProtocol的refer方法。

    前面章节也讲过,Dubbo默认提供了一系列Wrapper类对扩展实现类进行了功能增强,当然这里也不例外,Dubbo使用了ProtocolListenerWrapper / ProtocolFilterWrapper等类对DubboProtocol进行了功能增强。所以这里经过一次次调用后才调用到DubboProtocol的refer方法,DubboProtocol的refer代码内容如下:

    public <T> Invoker<T> refer(Class<T> serviceType , URL url) throws RpcException{
        // create rpc invoker.
        DubboInvoker<T> invoker = new DubboInvoker<T>(serviceType , url ,getClients(url) , invokers);
        invokers.add(invoker);
        return invoker;
    }

    如上代码,首先getClients方法从(12)到(18)创建服务消费端的NettyClient对象用来连接服务提供者。另外可知refer方法内部返回了一个DubboInvoker,这个就是原生的invoker对象,服务方远程服务转换就是为了这个invoker。代码(12)则对这个invoker进行装饰,使用一系列filter形成了责任链,invoker被放到责任链的末尾,下面看看buildInvokerChain如何形成责任链,代码如下:

    复制代码
    private static <T> Invoker<T> buildInvokerChain(final Invoker<T> invoker , String key , String group){
        Invoker<T> last = invoker;
        // 获取所有激活的filter,然后使用链表方式形成责任链
        List<Filter> filters = ExtensionLoader.getExtensionLoader(Filter.class).getActivateExtension(invoker.getUrl() , key , group);
        if(filters.size() > 0){
            for(int i=filters.size()-1 ; i>=0 ; i--){
                final Filter filter = filters.get(i);
                final Invoker<T> next = last;
                last = new Invoker<T>(){
                    public Class<T> getInterface(){
                        return invoker.getInterface();
                    }
                    public URL getUrl(){
                        return invoker.getUrl();
                    }
                    public boolean isAvailable(){
                        return invoker.isAvailable();
                    }
                    public Result invoke(Invocation invocation) throws RpcException{
                        return filter.invoke(next , invocation);    
                    }
                    public void destroy(){
                        invoker.destroy();
                    }
                    @Override
                    public String toString(){
                        return invoker.toString();
                    }
                };
            }
        }
        return last;
    }
    复制代码

    其中扩展接口Filter对应的实现类,如下所示:

    其中MonitorFilter和监控中心进行交互,FutureFilter用来实现异步调用,GenericFilter用来实现泛化调用,ActiveLimitFilter用来控制消费端最大并发调用量,ExecuteLimitFilter用来控制服务提供方最大并发处理量等,当然你可以写自己的filter。由于是责任链,所以ProtocolFilterWrapper的refer返回的是责任链头部的filter到ProtocolListenerWrapper,而ProtocolListenerWrapper的refer方法内容如下:

    public <T> Invoker<T> refer(Class<T> type , URL url) throws RpcException{
        ...
        return new ListenerInvokerWrapper<T>(protocol.refer(type,url) , Collections.unmodifiableList(ExtensionLoader.getExtensionLoader(InvokerListener.class).getActivateExtension(url , Constants.INVOKER_LISTENER_KEY)));
    }

    ProtocolListenerWrapper的refer返回的是ListenerInvokerWrapper对象,所以代码(9)返回的也是url对应的ListenerInvokerWrapper对象,然后再回到时序图代码(8)的toInvokers方法,可知toInvokers返回的是使用InvokerDelegete对ListenerInvokerWrapper包裹的对象。到这里RegistryDirectory里就维护了所有服务者的invoker列表,消费端发送信息时就是根据集群容错和负载均衡算法从invoker列表里选择一个进行调用,当服务提供者挂了的时候,Zookeeper会通知更新这invoker列表。

    到这里我们讲完了本节第一个时序图中的步骤(3),下面我们接着讲解步骤(4) 。

    默认下步骤(4)会调用FailoverCluster的join方法,FailoverCluster的join方法代码如下:

    public class FailoverCluster implements Cluster{
        private static final String NAME = "failover";
        public <T> Invoker<T> join(Directory<T> directory) throws RpcException{
            return new FailoverClusterInvoker<T>(directory);
        }
    }

    这里把directory对象包裹到了FailoverClusterInvoker,这里需要注意下directory就是上面讲解的RegistryDirectory,其内部维护了所有服务提供者的invoker列表,而FailoverCluster就是集群容错策略。

    其实Dubbo对cluster扩展接口实现类使用wrapper类MockClusterWrapper进行增强,这个从下图可以得到证明:

    实际上的调用时序图如下图所示:

    该时序图中(3)返回了FailbackClusterInvoker对象到(2),下面看看MockClusterWrapper的代码:

    复制代码
    public class MockClusterWrapper implements Cluster{
        private Cluster cluster;
        public MockClusterWrapper(Cluster cluster){
            this.cluster = cluster;
        }
        public <T> Invoker<T> join(Directory<T> directory) throws RpcException{
            return new MockClusterInvoker<T>(directory , this.cluster.join(directory));
        }
    }
    复制代码

    可知MockClusterWrapper类把FailoverClusterInvoker包装成了MockClusterInvoker实例,所以整个调用链最终调用返回的是MockClusterInvoker对象。也就是本文第一个时序图步骤(4)返回的是MockClusterWrapper。然后执行代码(13)获取MockClusterInvoker的代理实现invoker到客户端接口的转换,这里默认调用的是JavassistProxyFactory的getProxy方法,代码如下:

    public <T> T getProxy(Invoker<T> invoker , Class<?>[] interfaces){
        return (T)Proxy.getProxy(interfaces).newInstance(new InvokerInvocationHandler(invoker));
    }

    其中InvokerInvocationHandler为具体拦截器。至此我们按照逆序的方式把服务消费端启动流程讲解完毕,下面在顺过来讲解一次远程调用过程。

    Dubbo服务消费端一次调用流程原理分析

    同理先上时序图:

    由于服务消费端通过ReferenceConfig的get()方法返回的是一个代理类,并且方法拦截器为InvokerInvocationHandler,所以当调用了服务的方法后会被InvokerInvocationHandler拦截,执行如上时序图流程。

    如上流程,首先步骤(1)(2)(3)调用了集群容错策略FailoverClusterInvoker,其内部首先根据设置的负载均衡策略选择一个invoker作为FailoverClusterInvoker具体的远程调用者,如果调用发生异常,则根据FailoverClusterInvoker的策略重新选择一个invoker进行调用。

    FailoverClusterInvoker内每次调用具体invoker的invoke方法都会走到步骤(8)(9),然后步骤(10)(11)(12)是filter内创建的责任链,最后调用了原生的DubboInvoker,具体使用nettyclient与服务提供者进行交互。

      

  • 相关阅读:
    dedecms文章标题是在哪个数据库表?要批量替换关键词
    dedecms首页调用的简介一直修改不了是自动文章摘要在作怪
    如何进行微信营销?玩转微信公众平台营销理论手册
    用了那么久居然没发现firefox快捷键有如此多
    保护隐私:清除cookie、禁用cookie确保安全【分享给身边的朋友吧】
    如何更改firefox默认搜索引擎?一步搞定!
    微信红包中使用的技术:AA收款+随机算法
    马年添加一下马蹄印记统计(网站统计)
    今天上完就放假了,马年见
    三种dedecms调用相关文章的方法
  • 原文地址:https://www.cnblogs.com/cnndevelop/p/12186982.html
Copyright © 2011-2022 走看看