成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

dubbo源碼解析(四十四)服務暴露過程

light / 851人閱讀

摘要:服務暴露過程目標從源碼的角度分析服務暴露過程。導出服務,包含暴露服務到本地,和暴露服務到遠程兩個過程。其中服務暴露的第八步已經(jīng)沒有了。將泛化調用版本號或者等信息加入獲得服務暴露地址和端口號,利用內數(shù)據(jù)組裝成。

dubbo服務暴露過程
目標:從源碼的角度分析服務暴露過程。
前言

本來這一篇一個寫異步化改造的內容,但是最近我一直在想,某一部分的優(yōu)化改造該怎么去撰寫才能更加的讓讀者理解。我覺得還是需要先從整個調用鏈入手,先弄清楚了該功能在哪一個時機發(fā)生的,說通俗一點,這塊代碼是什么時候或者什么場景被執(zhí)行的,然后再去分析內部是如何實現(xiàn),最后闡述這樣改造的好處。

我在前面的文章都很少提及各個調用鏈的關系,各模塊之間也沒有串起來,并且要講解異步化改造我認為先弄懂服務的暴露和引用過程是非常有必要的,所以我將用兩片文章來講解服務暴露和服務引用的過程。

服務暴露過程

服務暴露過程大致可分為三個部分:

前置工作,主要用于檢查參數(shù),組裝 URL。

導出服務,包含暴露服務到本地 (JVM),和暴露服務到遠程兩個過程。

向注冊中心注冊服務,用于服務發(fā)現(xiàn)。

暴露起點

Spring中有一個ApplicationListener接口,其中定義了一個onApplicationEvent()方法,在當容器內發(fā)生任何事件時,此方法都會被觸發(fā)。

Dubbo中ServiceBean類實現(xiàn)了該接口,并且實現(xiàn)了onApplicationEvent方法:

@Override
public void onApplicationEvent(ContextRefreshedEvent event) {
      // 如果服務沒有被暴露并且服務沒有被取消暴露,則打印日志
    if (!isExported() && !isUnexported()) {
        if (logger.isInfoEnabled()) {
            logger.info("The service ready on spring started. service: " + getInterface());
        }
          // 導出
        export();
    }
}

只要服務沒有被暴露并且服務沒有被取消暴露,就暴露服務。執(zhí)行export方法。接下來就是跟官網(wǎng)的時序圖有關,對照著時序圖來看下面的過程。

我會在下面的小標題加上時序圖的每一步操作范圍,比如下面要講到的前置工作其實就是時序圖中的1:export(),那我會在標題括號里面寫1。

其中服務暴露的第八步已經(jīng)沒有了。

前置工作(1)

前置工作主要包含兩個部分,分別是配置檢查,以及 URL 裝配。在暴露服務之前,Dubbo 需要檢查用戶的配置是否合理,或者為用戶補充缺省配置。配置檢查完成后,接下來需要根據(jù)這些配置組裝 URL。在 Dubbo 中,URL 的作用十分重要。Dubbo 使用 URL 作為配置載體,所有的拓展點都是通過 URL 獲取配置。

配置檢查

在調用export方法之后,執(zhí)行的是ServiceConfig中的export方法。

public synchronized void export() {
    //檢查并且更新配置
    checkAndUpdateSubConfigs();

    // 如果不應該暴露,則直接結束
    if (!shouldExport()) {
        return;
    }

    // 如果使用延遲加載,則延遲delay時間后暴露服務
    if (shouldDelay()) {
        delayExportExecutor.schedule(this::doExport, delay, TimeUnit.MILLISECONDS);
    } else {
        // 暴露服務
        doExport();
    }
}

可以看到首先做的就是對配置的檢查和更新,執(zhí)行的是ServiceConfig中的checkAndUpdateSubConfigs方法。然后檢測是否應該暴露,如果不應該暴露,則直接結束,然后檢測是否配置了延遲加載,如果是,則使用定時器來實現(xiàn)延遲加載的目的。

checkAndUpdateSubConfigs()
public void checkAndUpdateSubConfigs() {
    // Use default configs defined explicitly on global configs
    // 用于檢測 provider、application 等核心配置類對象是否為空,
    // 若為空,則嘗試從其他配置類對象中獲取相應的實例。
    completeCompoundConfigs();
    // Config Center should always being started first.
    // 開啟配置中心
    startConfigCenter();
    // 檢測 provider 是否為空,為空則新建一個,并通過系統(tǒng)變量為其初始化
    checkDefault();
    // 檢查application是否為空
    checkApplication();
    // 檢查注冊中心是否為空
    checkRegistry();
    // 檢查protocols是否為空
    checkProtocol();
    this.refresh();
    // 核對元數(shù)據(jù)中心配置是否為空
    checkMetadataReport();

    // 服務接口名不能為空,否則拋出異常
    if (StringUtils.isEmpty(interfaceName)) {
        throw new IllegalStateException(" interface not allow null!");
    }

    // 檢測 ref 是否為泛化服務類型
    if (ref instanceof GenericService) {
        // 設置interfaceClass為GenericService
        interfaceClass = GenericService.class;
        if (StringUtils.isEmpty(generic)) {
            // 設置generic = true
            generic = Boolean.TRUE.toString();
        }
    } else {
        try {
            // 獲得接口類型
            interfaceClass = Class.forName(interfaceName, true, Thread.currentThread()
                    .getContextClassLoader());
        } catch (ClassNotFoundException e) {
            throw new IllegalStateException(e.getMessage(), e);
        }
        // 對 interfaceClass,以及  標簽中的必要字段進行檢查
        checkInterfaceAndMethods(interfaceClass, methods);
        // 對 ref 合法性進行檢測
        checkRef();
        generic = Boolean.FALSE.toString();
    }
    // stub local一樣都是配置本地存根
    if (local != null) {
        if ("true".equals(local)) {
            local = interfaceName + "Local";
        }
        Class localClass;
        try {
            localClass = ClassHelper.forNameWithThreadContextClassLoader(local);
        } catch (ClassNotFoundException e) {
            throw new IllegalStateException(e.getMessage(), e);
        }
        if (!interfaceClass.isAssignableFrom(localClass)) {
            throw new IllegalStateException("The local implementation class " + localClass.getName() + " not implement interface " + interfaceName);
        }
    }
    if (stub != null) {
        if ("true".equals(stub)) {
            stub = interfaceName + "Stub";
        }
        Class stubClass;
        try {
            stubClass = ClassHelper.forNameWithThreadContextClassLoader(stub);
        } catch (ClassNotFoundException e) {
            throw new IllegalStateException(e.getMessage(), e);
        }
        if (!interfaceClass.isAssignableFrom(stubClass)) {
            throw new IllegalStateException("The stub implementation class " + stubClass.getName() + " not implement interface " + interfaceName);
        }
    }
    // 本地存根合法性校驗
    checkStubAndLocal(interfaceClass);
    // mock合法性校驗
    checkMock(interfaceClass);
}

可以看到,該方法中是對各類配置的校驗,并且更新部分配置。其中檢查的細節(jié)我就不展開,因為服務暴露整個過程才是本文重點。

在經(jīng)過shouldExport()和shouldDelay()兩個方法檢測后,會執(zhí)行ServiceConfig的doExport()方法

doExport()
protected synchronized void doExport() {
    // 如果調用不暴露的方法,則unexported值為true
    if (unexported) {
        throw new IllegalStateException("The service " + interfaceClass.getName() + " has already unexported!");
    }
    // 如果服務已經(jīng)暴露了,則直接結束
    if (exported) {
        return;
    }
    // 設置已經(jīng)暴露
    exported = true;

    // 如果path為空,則賦值接口名稱
    if (StringUtils.isEmpty(path)) {
        path = interfaceName;
    }
    // 多協(xié)議多注冊中心暴露服務
    doExportUrls();
}

該方法就是對于服務是否暴露在一次校驗,然后會執(zhí)行ServiceConfig的doExportUrls()方法,對于多協(xié)議多注冊中心暴露服務進行支持。

doExportUrls()
private void doExportUrls() {
    // 加載注冊中心鏈接
    List registryURLs = loadRegistries(true);
    // 遍歷 protocols,并在每個協(xié)議下暴露服務
    for (ProtocolConfig protocolConfig : protocols) {
        // 以path、group、version來作為服務唯一性確定的key
        String pathKey = URL.buildKey(getContextPath(protocolConfig).map(p -> p + "/" + path).orElse(path), group, version);
        ProviderModel providerModel = new ProviderModel(pathKey, ref, interfaceClass);
        ApplicationModel.initProviderModel(pathKey, providerModel);
        // 組裝 URL
        doExportUrlsFor1Protocol(protocolConfig, registryURLs);
    }
}

從該方法可以看到:

loadRegistries()方法是加載注冊中心鏈接。

服務的唯一性是通過path、group、version一起確定的。

doExportUrlsFor1Protocol()方法開始組裝URL。

loadRegistries()
protected List loadRegistries(boolean provider) {
    // check && override if necessary
    List registryList = new ArrayList();
    // 如果registries為空,直接返回空集合
    if (CollectionUtils.isNotEmpty(registries)) {
        // 遍歷注冊中心配置集合registries
        for (RegistryConfig config : registries) {
            // 獲得地址
            String address = config.getAddress();
            // 若地址為空,則設置為0.0.0.0
            if (StringUtils.isEmpty(address)) {
                address = Constants.ANYHOST_VALUE;
            }
            // 如果地址為N/A,則跳過
            if (!RegistryConfig.NO_AVAILABLE.equalsIgnoreCase(address)) {
                Map map = new HashMap();
                // 添加 ApplicationConfig 中的字段信息到 map 中
                appendParameters(map, application);
                // 添加 RegistryConfig 字段信息到 map 中
                appendParameters(map, config);
                // 添加path
                map.put(Constants.PATH_KEY, RegistryService.class.getName());
                // 添加 協(xié)議版本、發(fā)布版本,時間戳 等信息到 map 中
                appendRuntimeParameters(map);
                // 如果map中沒有protocol,則默認為使用dubbo協(xié)議
                if (!map.containsKey(Constants.PROTOCOL_KEY)) {
                    map.put(Constants.PROTOCOL_KEY, Constants.DUBBO_PROTOCOL);
                }
                // 解析得到 URL 列表,address 可能包含多個注冊中心 ip,因此解析得到的是一個 URL 列表
                List urls = UrlUtils.parseURLs(address, map);

                // 遍歷URL 列表
                for (URL url : urls) {
                    // 將 URL 協(xié)議頭設置為 registry
                    url = URLBuilder.from(url)
                            .addParameter(Constants.REGISTRY_KEY, url.getProtocol())
                            .setProtocol(Constants.REGISTRY_PROTOCOL)
                            .build();
                    // 通過判斷條件,決定是否添加 url 到 registryList 中,條件如下:
                    // 如果是服務提供者,并且是注冊中心服務   或者   是消費者端,并且是訂閱服務
                    // 則加入到registryList
                    if ((provider && url.getParameter(Constants.REGISTER_KEY, true))
                            || (!provider && url.getParameter(Constants.SUBSCRIBE_KEY, true))) {
                        registryList.add(url);
                    }
                }
            }
        }
    }
    return registryList;
}
組裝URL

我在以前的文章也提到過,dubbo內部用URL來攜帶各類配置,貫穿整個調用鏈,它就是配置的載體。服務的配置被組裝到URL中就是從這里開始,上面提到遍歷每個協(xié)議配置,在每個協(xié)議下都暴露服務,就會執(zhí)行ServiceConfig的doExportUrlsFor1Protocol()方法,該方法前半部分實現(xiàn)了組裝URL的邏輯,后半部分實現(xiàn)了暴露dubbo服務等邏輯,其中為用分割線分隔了。

doExportUrlsFor1Protocol()
private void doExportUrlsFor1Protocol(ProtocolConfig protocolConfig, List registryURLs) {
    // 獲取協(xié)議名
    String name = protocolConfig.getName();
    // 如果為空,則是默認的dubbo
    if (StringUtils.isEmpty(name)) {
        name = Constants.DUBBO;
    }

    Map map = new HashMap();
    // 設置服務提供者冊
    map.put(Constants.SIDE_KEY, Constants.PROVIDER_SIDE);

    // 添加 協(xié)議版本、發(fā)布版本,時間戳 等信息到 map 中
    appendRuntimeParameters(map);
    // 添加metrics、application、module、provider、protocol的所有信息到map
    appendParameters(map, metrics);
    appendParameters(map, application);
    appendParameters(map, module);
    appendParameters(map, provider, Constants.DEFAULT_KEY);
    appendParameters(map, protocolConfig);
    appendParameters(map, this);
    // 如果method的配置列表不為空
    if (CollectionUtils.isNotEmpty(methods)) {
        // 遍歷method配置列表
        for (MethodConfig method : methods) {
            // 把方法名加入map
            appendParameters(map, method, method.getName());
            // 添加 MethodConfig 對象的字段信息到 map 中,鍵 = 方法名.屬性名。
            // 比如存儲  對應的 MethodConfig,
            // 鍵 = sayHello.retries,map = {"sayHello.retries": 2, "xxx": "yyy"}
            String retryKey = method.getName() + ".retry";
            if (map.containsKey(retryKey)) {
                String retryValue = map.remove(retryKey);
                // 如果retryValue為false,則不重試,設置值為0
                if ("false".equals(retryValue)) {
                    map.put(method.getName() + ".retries", "0");
                }
            }
            // 獲得ArgumentConfig列表
            List arguments = method.getArguments();
            if (CollectionUtils.isNotEmpty(arguments)) {
                // 遍歷ArgumentConfig列表
                for (ArgumentConfig argument : arguments) {
                    // convert argument type
                    // // 檢測 type 屬性是否為空,或者空串
                    if (argument.getType() != null && argument.getType().length() > 0) {
                        // 利用反射獲取該服務的所有方法集合
                        Method[] methods = interfaceClass.getMethods();
                        // visit all methods
                        if (methods != null && methods.length > 0) {
                            // 遍歷所有方法
                            for (int i = 0; i < methods.length; i++) {
                                // 獲得方法名
                                String methodName = methods[i].getName();
                                // target the method, and get its signature
                                // 找到目標方法
                                if (methodName.equals(method.getName())) {
                                    // 通過反射獲取目標方法的參數(shù)類型數(shù)組 argtypes
                                    Class[] argtypes = methods[i].getParameterTypes();
                                    // one callback in the method
                                    // 如果下標為-1
                                    if (argument.getIndex() != -1) {
                                        // 檢測 argType 的名稱與 ArgumentConfig 中的 type 屬性是否一致
                                        if (argtypes[argument.getIndex()].getName().equals(argument.getType())) {
                                            //  添加 ArgumentConfig 字段信息到 map 中
                                            appendParameters(map, argument, method.getName() + "." + argument.getIndex());
                                        } else {
                                            // 不一致,則拋出異常
                                            throw new IllegalArgumentException("Argument config error : the index attribute and type attribute not match :index :" + argument.getIndex() + ", type:" + argument.getType());
                                        }
                                    } else {
                                        // multiple callbacks in the method
                                        // 遍歷參數(shù)類型數(shù)組 argtypes,查找 argument.type 類型的參數(shù)
                                        for (int j = 0; j < argtypes.length; j++) {
                                            Class argclazz = argtypes[j];
                                            if (argclazz.getName().equals(argument.getType())) {
                                                // 如果找到,則添加 ArgumentConfig 字段信息到 map 中
                                                appendParameters(map, argument, method.getName() + "." + j);
                                                if (argument.getIndex() != -1 && argument.getIndex() != j) {
                                                    throw new IllegalArgumentException("Argument config error : the index attribute and type attribute not match :index :" + argument.getIndex() + ", type:" + argument.getType());
                                                }
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    } else if (argument.getIndex() != -1) {
                        // 用戶未配置 type 屬性,但配置了 index 屬性,且 index != -1,則直接添加到map
                        appendParameters(map, argument, method.getName() + "." + argument.getIndex());
                    } else {
                        // 拋出異常
                        throw new IllegalArgumentException("Argument config must set index or type attribute.eg:  or ");
                    }

                }
            }
        } // end of methods for
    }

    // 如果是泛化調用,則在map中設置generic和methods
    if (ProtocolUtils.isGeneric(generic)) {
        map.put(Constants.GENERIC_KEY, generic);
        map.put(Constants.METHODS_KEY, Constants.ANY_VALUE);
    } else {
        // 獲得版本號
        String revision = Version.getVersion(interfaceClass, version);
        // 放入map
        if (revision != null && revision.length() > 0) {
            map.put(Constants.REVISION_KEY, revision);
        }

        // 獲得方法集合
        String[] methods = Wrapper.getWrapper(interfaceClass).getMethodNames();
        // 如果為空,則告警
        if (methods.length == 0) {
            logger.warn("No method found in service interface " + interfaceClass.getName());
            // 設置method為*
            map.put(Constants.METHODS_KEY, Constants.ANY_VALUE);
        } else {
            // 否則加入方法集合
            map.put(Constants.METHODS_KEY, StringUtils.join(new HashSet(Arrays.asList(methods)), ","));
        }
    }
    // 把token 的值加入到map中
    if (!ConfigUtils.isEmpty(token)) {
        if (ConfigUtils.isDefault(token)) {
            map.put(Constants.TOKEN_KEY, UUID.randomUUID().toString());
        } else {
            map.put(Constants.TOKEN_KEY, token);
        }
    }
    // export service
    // 獲得地址
    String host = this.findConfigedHosts(protocolConfig, registryURLs, map);
    // 獲得端口號
    Integer port = this.findConfigedPorts(protocolConfig, name, map);
    // 生成 URL
    URL url = new URL(name, host, port, getContextPath(protocolConfig).map(p -> p + "/" + path).orElse(path), map);

    // —————————————————————————————————————分割線———————————————————————————————————————


    // 加載 ConfiguratorFactory,并生成 Configurator 實例,判斷是否有該協(xié)議的實現(xiàn)存在
    if (ExtensionLoader.getExtensionLoader(ConfiguratorFactory.class)
            .hasExtension(url.getProtocol())) {
        // 通過實例配置 url
        url = ExtensionLoader.getExtensionLoader(ConfiguratorFactory.class)
                .getExtension(url.getProtocol()).getConfigurator(url).configure(url);
    }

    String scope = url.getParameter(Constants.SCOPE_KEY);
    // don"t export when none is configured
    // // 如果 scope = none,則什么都不做
    if (!Constants.SCOPE_NONE.equalsIgnoreCase(scope)) {

        // export to local if the config is not remote (export to remote only when config is remote)
        // // scope != remote,暴露到本地
        if (!Constants.SCOPE_REMOTE.equalsIgnoreCase(scope)) {
            // 暴露到本地
            exportLocal(url);
        }
        // export to remote if the config is not local (export to local only when config is local)
        // // scope != local,導出到遠程
        if (!Constants.SCOPE_LOCAL.equalsIgnoreCase(scope)) {
            if (logger.isInfoEnabled()) {
                logger.info("Export dubbo service " + interfaceClass.getName() + " to url " + url);
            }
            // 如果注冊中心鏈接集合不為空
            if (CollectionUtils.isNotEmpty(registryURLs)) {
                // 遍歷注冊中心
                for (URL registryURL : registryURLs) {
                    // 添加dynamic配置
                    url = url.addParameterIfAbsent(Constants.DYNAMIC_KEY, registryURL.getParameter(Constants.DYNAMIC_KEY));
                    // 加載監(jiān)視器鏈接
                    URL monitorUrl = loadMonitor(registryURL);
                    if (monitorUrl != null) {
                        // 添加監(jiān)視器配置
                        url = url.addParameterAndEncoded(Constants.MONITOR_KEY, monitorUrl.toFullString());
                    }
                    if (logger.isInfoEnabled()) {
                        logger.info("Register dubbo service " + interfaceClass.getName() + " url " + url + " to registry " + registryURL);
                    }

                    // For providers, this is used to enable custom proxy to generate invoker
                    // 獲得代理方式
                    String proxy = url.getParameter(Constants.PROXY_KEY);
                    if (StringUtils.isNotEmpty(proxy)) {
                        // 添加代理方式到注冊中心到url
                        registryURL = registryURL.addParameter(Constants.PROXY_KEY, proxy);
                    }

                    // 為服務提供類(ref)生成 Invoker
                    Invoker invoker = proxyFactory.getInvoker(ref, (Class) interfaceClass, registryURL.addParameterAndEncoded(Constants.EXPORT_KEY, url.toFullString()));
                    // DelegateProviderMetaDataInvoker 用于持有 Invoker 和 ServiceConfig
                    DelegateProviderMetaDataInvoker wrapperInvoker = new DelegateProviderMetaDataInvoker(invoker, this);

                    // 暴露服務,并且生成Exporter
                    Exporter exporter = protocol.export(wrapperInvoker);
                    // 加入到暴露者集合中
                    exporters.add(exporter);
                }
            } else {
                // 不存在注冊中心,則僅僅暴露服務,不會記錄暴露到地址
                Invoker invoker = proxyFactory.getInvoker(ref, (Class) interfaceClass, url);
                DelegateProviderMetaDataInvoker wrapperInvoker = new DelegateProviderMetaDataInvoker(invoker, this);

                Exporter exporter = protocol.export(wrapperInvoker);
                exporters.add(exporter);
            }
            /**
             * @since 2.7.0
             * ServiceData Store
             */
            MetadataReportService metadataReportService = null;
            // 如果元數(shù)據(jù)中心服務不為空,則發(fā)布該服務,也就是在元數(shù)據(jù)中心記錄url中到部分配置
            if ((metadataReportService = getMetadataReportService()) != null) {
                metadataReportService.publishProvider(url);
            }
        }
    }
    this.urls.add(url);
}

先看分割線上面部分,就是組裝URL的全過程,我覺得大致可以分為一下步驟:

它把metrics、application、module、provider、protocol等所有配置都放入map中,

針對method都配置,先做簽名校驗,先找到該服務是否有配置的方法存在,然后該方法簽名是否有這個參數(shù)存在,都核對成功才將method的配置加入map。

將泛化調用、版本號、method或者methods、token等信息加入map

獲得服務暴露地址和端口號,利用map內數(shù)據(jù)組裝成URL。

創(chuàng)建invoker(2,3)

暴露到遠程的源碼直接看doExportUrlsFor1Protocol()方法分割線下半部分。當生成暴露者的時候,服務已經(jīng)暴露,接下來會細致的分析這暴露內部的過程??梢园l(fā)現(xiàn)無論暴露到本地還是遠程,都會通過代理工廠創(chuàng)建invoker。這個時候就走到了上述時序圖的ProxyFactory。我在這篇文章中有講到invoker:dubbo源碼解析(十九)遠程調用——開篇,首先我們來看看invoker如何誕生的。Invoker 是由 ProxyFactory 創(chuàng)建而來,Dubbo 默認的 ProxyFactory 實現(xiàn)類是 JavassistProxyFactory。JavassistProxyFactory中有一個getInvoker()方法。

獲取invoker方法
getInvoker()
public  Invoker getInvoker(T proxy, Class type, URL url) {
    // TODO Wrapper cannot handle this scenario correctly: the classname contains "$"
    // // 為目標類創(chuàng)建 Wrapper
    final Wrapper wrapper = Wrapper.getWrapper(proxy.getClass().getName().indexOf("$") < 0 ? proxy.getClass() : type);
    // 創(chuàng)建匿名 Invoker 類對象,并實現(xiàn) doInvoke 方法。
    return new AbstractProxyInvoker(proxy, type, url) {
        @Override
        protected Object doInvoke(T proxy, String methodName,
                                  Class[] parameterTypes,
                                  Object[] arguments) throws Throwable {
            // 調用 Wrapper 的 invokeMethod 方法,invokeMethod 最終會調用目標方法
            return wrapper.invokeMethod(proxy, methodName, parameterTypes, arguments);
        }
    };
}

可以看到,該方法就是創(chuàng)建了一個匿名的Invoker類對象,在doInvoke()方法中調用wrapper.invokeMethod()方法。Wrapper 是一個抽象類,僅可通過 getWrapper(Class) 方法創(chuàng)建子類。在創(chuàng)建 Wrapper 子類的過程中,子類代碼生成邏輯會對 getWrapper 方法傳入的 Class 對象進行解析,拿到諸如類方法,類成員變量等信息。以及生成 invokeMethod 方法代碼和其他一些方法代碼。代碼生成完畢后,通過 Javassist 生成 Class 對象,最后再通過反射創(chuàng)建 Wrapper 實例。那么我們先來看看getWrapper()方法:

getWrapper()
public static Wrapper getWrapper(Class c) {
    while (ClassGenerator.isDynamicClass(c)) // can not wrapper on dynamic class.
    {
        // 返回該對象的超類
        c = c.getSuperclass();
    }

    // 如果超類就是Object,則返回子類Wrapper
    if (c == Object.class) {
        return OBJECT_WRAPPER;
    }

    // 從緩存中獲取 Wrapper 實例
    Wrapper ret = WRAPPER_MAP.get(c);
    // 如果沒有命中,則創(chuàng)建 Wrapper
    if (ret == null) {
        // 創(chuàng)建Wrapper
        ret = makeWrapper(c);
        // 寫入緩存
        WRAPPER_MAP.put(c, ret);
    }
    return ret;
}

該方法只是對Wrapper 做了緩存。主要的邏輯在makeWrapper()。

makeWrapper()
    // 檢測 c 是否為基本類型,若是則拋出異常
    if (c.isPrimitive()) {
        throw new IllegalArgumentException("Can not create wrapper for primitive type: " + c);
    }

    // 獲得類名
    String name = c.getName();
    // 獲得類加載器
    ClassLoader cl = ClassHelper.getClassLoader(c);

    // c1 用于存儲 setPropertyValue 方法代碼
    StringBuilder c1 = new StringBuilder("public void setPropertyValue(Object o, String n, Object v){ ");
    // c2 用于存儲 getPropertyValue 方法代碼
    StringBuilder c2 = new StringBuilder("public Object getPropertyValue(Object o, String n){ ");
    // c3 用于存儲 invokeMethod 方法代碼
    StringBuilder c3 = new StringBuilder("public Object invokeMethod(Object o, String n, Class[] p, Object[] v) throws " + InvocationTargetException.class.getName() + "{ ");

    // 生成類型轉換代碼及異常捕捉代碼,比如:
    //   DemoService w; try { w = ((DemoServcie) $1); }}catch(Throwable e){ throw new IllegalArgumentException(e); }
    c1.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");
    c2.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");
    c3.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");

    // pts 用于存儲成員變量名和類型
    Map> pts = new HashMap<>(); // 
    // ms 用于存儲方法描述信息(可理解為方法簽名)及 Method 實例
    Map ms = new LinkedHashMap<>(); // 
    // mns 為方法名列表
    List mns = new ArrayList<>(); // method names.
    // dmns 用于存儲“定義在當前類中的方法”的名稱
    List dmns = new ArrayList<>(); // declaring method names.

    // get all public field.
    // 獲取 public 訪問級別的字段,并為所有字段生成條件判斷語句
    for (Field f : c.getFields()) {
        String fn = f.getName();
        Class ft = f.getType();
        if (Modifier.isStatic(f.getModifiers()) || Modifier.isTransient(f.getModifiers())) {
            // 忽略關鍵字 static 或 transient 修飾的變量
            continue;
        }
        // 生成條件判斷及賦值語句,比如:
        // if( $2.equals("name") ) { w.name = (java.lang.String) $3; return;}
        // if( $2.equals("age") ) { w.age = ((Number) $3).intValue(); return;}
        c1.append(" if( $2.equals("").append(fn).append("") ){ w.").append(fn).append("=").append(arg(ft, "$3")).append("; return; }");
        // 生成條件判斷及返回語句,比如:
        // if( $2.equals("name") ) { return ($w)w.name; }
        c2.append(" if( $2.equals("").append(fn).append("") ){ return ($w)w.").append(fn).append("; }");
        // 存儲 <字段名, 字段類型> 鍵值對到 pts 中
        pts.put(fn, ft);
    }

    // 獲得c類的所有方法
    Method[] methods = c.getMethods();
    // get all public method.
    // 檢測 c 中是否包含在當前類中聲明的方法
    boolean hasMethod = hasMethods(methods);
    // 如果包含
    if (hasMethod) {
        c3.append(" try{");
        for (Method m : methods) {
            //ignore Object"s method.
            // 忽略 Object 中定義的方法
            if (m.getDeclaringClass() == Object.class) {
                continue;
            }

            // 獲得方法的名稱
            String mn = m.getName();
            // 生成方法名判斷語句,比如:
            // if ( "sayHello".equals( $2 )
            c3.append(" if( "").append(mn).append("".equals( $2 ) ");
            int len = m.getParameterTypes().length;
            // 生成“運行時傳入的參數(shù)數(shù)量與方法參數(shù)列表長度”判斷語句,比如:
            // && $3.length == 2
            c3.append(" && ").append(" $3.length == ").append(len);

            boolean override = false;
            for (Method m2 : methods) {
                // 檢測方法是否存在重載情況,條件為:方法對象不同 && 方法名相同
                if (m != m2 && m.getName().equals(m2.getName())) {
                    override = true;
                    break;
                }
            }
            // 對重載方法進行處理,考慮下面的方法:
            //    1. void sayHello(Integer, String)
            //    2. void sayHello(Integer, Integer)
            // 方法名相同,參數(shù)列表長度也相同,因此不能僅通過這兩項判斷兩個方法是否相等。
            // 需要進一步判斷方法的參數(shù)類型
            if (override) {
                if (len > 0) {
                    for (int l = 0; l < len; l++) {
                        c3.append(" && ").append(" $3[").append(l).append("].getName().equals("")
                                .append(m.getParameterTypes()[l].getName()).append("")");
                    }
                }
            }
            // 添加 ) {,完成方法判斷語句,此時生成的代碼可能如下(已格式化):
            // if ("sayHello".equals($2)
            //     && $3.length == 2
            //     && $3[0].getName().equals("java.lang.Integer")
            //     && $3[1].getName().equals("java.lang.String")) {
            c3.append(" ) { ");

            // 根據(jù)返回值類型生成目標方法調用語句
            if (m.getReturnType() == Void.TYPE) {
                // w.sayHello((java.lang.Integer)$4[0], (java.lang.String)$4[1]); return null;
                c3.append(" w.").append(mn).append("(").append(args(m.getParameterTypes(), "$4")).append(");").append(" return null;");
            } else {
                // return w.sayHello((java.lang.Integer)$4[0], (java.lang.String)$4[1]);
                c3.append(" return ($w)w.").append(mn).append("(").append(args(m.getParameterTypes(), "$4")).append(");");
            }

            c3.append(" }");

            // 添加方法名到 mns 集合中
            mns.add(mn);
            // 檢測當前方法是否在 c 中被聲明的
            if (m.getDeclaringClass() == c) {
                // 若是,則將當前方法名添加到 dmns 中
                dmns.add(mn);
            }
            ms.put(ReflectUtils.getDesc(m), m);
        }
        // 添加異常捕捉語句
        c3.append(" } catch(Throwable e) { ");
        c3.append("     throw new java.lang.reflect.InvocationTargetException(e); ");
        c3.append(" }");
    }

    // 添加NoSuchMethodException異常
    c3.append(" throw new " + NoSuchMethodException.class.getName() + "("Not found method ""+$2+"" in class " + c.getName() + "."); }");

    // deal with get/set method.
    Matcher matcher;
    // 處理 get/set 方法
    for (Map.Entry entry : ms.entrySet()) {
        // 獲得方法名稱
        String md = entry.getKey();
        // 獲得Method方法
        Method method = entry.getValue();
        // 如果是get開頭的方法
        if ((matcher = ReflectUtils.GETTER_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            // 獲取屬性名
            String pn = propertyName(matcher.group(1));
            // 生成屬性判斷以及返回語句,示例如下:
            // if( $2.equals("name") ) { return ($w).w.getName(); }
            c2.append(" if( $2.equals("").append(pn).append("") ){ return ($w)w.").append(method.getName()).append("(); }");
            pts.put(pn, method.getReturnType());
        } else if ((matcher = ReflectUtils.IS_HAS_CAN_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            String pn = propertyName(matcher.group(1));
            c2.append(" if( $2.equals("").append(pn).append("") ){ return ($w)w.").append(method.getName()).append("(); }");
            // 存儲屬性名和返回類型到pts
            pts.put(pn, method.getReturnType());
        } else if ((matcher = ReflectUtils.SETTER_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            // 如果是set開頭的方法
            // 獲得參數(shù)類型
            Class pt = method.getParameterTypes()[0];
            // 獲得屬性名
            String pn = propertyName(matcher.group(1));
            // 生成屬性判斷以及 setter 調用語句,示例如下:
            // if( $2.equals("name") ) { w.setName((java.lang.String)$3); return; }
            c1.append(" if( $2.equals("").append(pn).append("") ){ w.").append(method.getName()).append("(").append(arg(pt, "$3")).append("); return; }");
            pts.put(pn, pt);
        }
    }
    // 添加 NoSuchPropertyException 異常拋出代碼
    c1.append(" throw new " + NoSuchPropertyException.class.getName() + "("Not found property ""+$2+"" field or setter method in class " + c.getName() + "."); }");
    c2.append(" throw new " + NoSuchPropertyException.class.getName() + "("Not found property ""+$2+"" field or setter method in class " + c.getName() + "."); }");

    // make class
    long id = WRAPPER_CLASS_COUNTER.getAndIncrement();
    // 創(chuàng)建類生成器
    ClassGenerator cc = ClassGenerator.newInstance(cl);
    // 設置類名及超類
    cc.setClassName((Modifier.isPublic(c.getModifiers()) ? Wrapper.class.getName() : c.getName() + "$sw") + id);
    cc.setSuperClass(Wrapper.class);

    // 添加默認的構造函數(shù)
    cc.addDefaultConstructor();
    // 添加字段
    cc.addField("public static String[] pns;"); // property name array.
    cc.addField("public static " + Map.class.getName() + " pts;"); // property type map.
    cc.addField("public static String[] mns;"); // all method name array.
    cc.addField("public static String[] dmns;"); // declared method name array.
    for (int i = 0, len = ms.size(); i < len; i++) {
        cc.addField("public static Class[] mts" + i + ";");
    }

    // 添加方法
    cc.addMethod("public String[] getPropertyNames(){ return pns; }");
    cc.addMethod("public boolean hasProperty(String n){ return pts.containsKey($1); }");
    cc.addMethod("public Class getPropertyType(String n){ return (Class)pts.get($1); }");
    cc.addMethod("public String[] getMethodNames(){ return mns; }");
    cc.addMethod("public String[] getDeclaredMethodNames(){ return dmns; }");
    cc.addMethod(c1.toString());
    cc.addMethod(c2.toString());
    cc.addMethod(c3.toString());

    try {
        // 生成類
        Class wc = cc.toClass();
        // setup static field.
        // 設置字段值
        wc.getField("pts").set(null, pts);
        wc.getField("pns").set(null, pts.keySet().toArray(new String[0]));
        wc.getField("mns").set(null, mns.toArray(new String[0]));
        wc.getField("dmns").set(null, dmns.toArray(new String[0]));
        int ix = 0;
        for (Method m : ms.values()) {
            wc.getField("mts" + ix++).set(null, m.getParameterTypes());
        }
        // 創(chuàng)建 Wrapper 實例
        return (Wrapper) wc.newInstance();
    } catch (RuntimeException e) {
        throw e;
    } catch (Throwable e) {
        throw new RuntimeException(e.getMessage(), e);
    } finally {
        cc.release();
        ms.clear();
        mns.clear();
        dmns.clear();
    }
}

該方法有點長,大致可以分為幾個步驟:

初始化了c1、c2、c3、pts、ms、mns、dmns變量,向 c1、c2、c3 中添加方法定義和類型轉換代碼。

為 public 級別的字段生成條件判斷取值與賦值代碼

為定義在當前類中的方法生成判斷語句,和方法調用語句。

處理 getter、setter 以及以 is/has/can 開頭的方法。處理方式是通過正則表達式獲取方法類型(get/set/is/...),以及屬性名。之后為屬性名生成判斷語句,然后為方法生成調用語句。

通過 ClassGenerator 為剛剛生成的代碼構建 Class 類,并通過反射創(chuàng)建對象。ClassGenerator 是 Dubbo 自己封裝的,該類的核心是 toClass() 的重載方法 toClass(ClassLoader, ProtectionDomain),該方法通過 javassist 構建 Class。

服務暴露

服務暴露分為暴露到本地 (JVM),和暴露到遠程。doExportUrlsFor1Protocol()方法分割線下半部分就是服務暴露的邏輯。根據(jù)scope的配置分為:

scope = none,不暴露服務

scope != remote,暴露到本地

scope != local,暴露到遠程

暴露到本地

導出本地執(zhí)行的是ServiceConfig中的exportLocal()方法。

exportLocal()(4)
private void exportLocal(URL url) {
    // 如果協(xié)議不是injvm
    if (!Constants.LOCAL_PROTOCOL.equalsIgnoreCase(url.getProtocol())) {
        // 生成本地的url,分別把協(xié)議改為injvm,設置host和port
        URL local = URLBuilder.from(url)
                .setProtocol(Constants.LOCAL_PROTOCOL)
                .setHost(LOCALHOST_VALUE)
                .setPort(0)
                .build();
        // 通過代理工程創(chuàng)建invoker
        // 再調用export方法進行暴露服務,生成Exporter
        Exporter exporter = protocol.export(
                proxyFactory.getInvoker(ref, (Class) interfaceClass, local));
        // 把生成的暴露者加入集合
        exporters.add(exporter);
        logger.info("Export dubbo service " + interfaceClass.getName() + " to local registry");
    }
}

本地暴露調用的是injvm協(xié)議方法,也就是InjvmProtocol 的 export()方法。

export()(5)
public  Exporter export(Invoker invoker) throws RpcException {
    return new InjvmExporter(invoker, invoker.getUrl().getServiceKey(), exporterMap);
}

該方法只是創(chuàng)建了一個,因為暴露到本地,所以在同一個jvm中。所以不需要其他操作。

暴露到遠程

暴露到遠程的邏輯要比本地復雜的多,它大致可以分為服務暴露和服務注冊兩個過程。先來看看服務暴露。我們知道dubbo有很多協(xié)議實現(xiàn),在doExportUrlsFor1Protocol()方法分割線下半部分中,生成了Invoker后,就需要調用protocol 的 export()方法,很多人會認為這里的export()就是配置中指定的協(xié)議實現(xiàn)中的方法,但這里是不對的。因為暴露到遠程后需要進行服務注冊,而RegistryProtocol的 export()方法就是實現(xiàn)了服務暴露和服務注冊兩個過程。所以這里的export()調用的是RegistryProtocol的 export()。

export()
public  Exporter export(final Invoker originInvoker) throws RpcException {
    // 獲得注冊中心的url
    URL registryUrl = getRegistryUrl(originInvoker);
    // url to export locally
    //獲得已經(jīng)注冊的服務提供者url
    URL providerUrl = getProviderUrl(originInvoker);

    // Subscribe the override data
    // FIXME When the provider subscribes, it will affect the scene : a certain JVM exposes the service and call
    //  the same service. Because the subscribed is cached key with the name of the service, it causes the
    //  subscription information to cover.
    // 獲取override訂閱 URL
    final URL overrideSubscribeUrl = getSubscribedOverrideUrl(providerUrl);
    // 創(chuàng)建override的監(jiān)聽器
    final OverrideListener overrideSubscribeListener = new OverrideListener(overrideSubscribeUrl, originInvoker);
    // 把監(jiān)聽器添加到集合
    overrideListeners.put(overrideSubscribeUrl, overrideSubscribeListener);

    // 根據(jù)override的配置來覆蓋原來的url,使得配置是最新的。
    providerUrl = overrideUrlWithConfig(providerUrl, overrideSubscribeListener);
    //export invoker
    // 服務暴露
    final ExporterChangeableWrapper exporter = doLocalExport(originInvoker, providerUrl);

    // url to registry
    // 根據(jù) URL 加載 Registry 實現(xiàn)類,比如ZookeeperRegistry
    final Registry registry = getRegistry(originInvoker);
    // 返回注冊到注冊表的url并過濾url參數(shù)一次
    final URL registeredProviderUrl = getRegisteredProviderUrl(providerUrl, registryUrl);
    // 生成ProviderInvokerWrapper,它會保存服務提供方和消費方的調用地址和代理對象
    ProviderInvokerWrapper providerInvokerWrapper = ProviderConsumerRegTable.registerProvider(originInvoker,
            registryUrl, registeredProviderUrl);


    // ————————————————————————————————分割線——————————————————————————————————————
    //to judge if we need to delay publish
    // 獲取 register 參數(shù)
    boolean register = registeredProviderUrl.getParameter("register", true);
    // 如果需要注冊服務
    if (register) {
        // 向注冊中心注冊服務
        register(registryUrl, registeredProviderUrl);
        // 設置reg為true,表示服務注冊
        providerInvokerWrapper.setReg(true);
    }

    // Deprecated! Subscribe to override rules in 2.6.x or before.
    // 向注冊中心進行訂閱 override 數(shù)據(jù)
    registry.subscribe(overrideSubscribeUrl, overrideSubscribeListener);

    // 設置注冊中心url
    exporter.setRegisterUrl(registeredProviderUrl);
    // 設置override數(shù)據(jù)訂閱的url
    exporter.setSubscribeUrl(overrideSubscribeUrl);
    //Ensure that a new exporter instance is returned every time export
    // 創(chuàng)建并返回 DestroyableExporter
    return new DestroyableExporter<>(exporter);
}

從代碼上看,我用分割線分成兩部分,分別是服務暴露和服務注冊。該方法的邏輯大致分為以下幾個步驟:

獲得服務提供者的url,再通過override數(shù)據(jù)重新配置url,然后執(zhí)行doLocalExport()進行服務暴露。

加載注冊中心實現(xiàn)類,向注冊中心注冊服務。

向注冊中心進行訂閱 override 數(shù)據(jù)。

創(chuàng)建并返回 DestroyableExporter

服務暴露先調用的是RegistryProtocol的doLocalExport()方法

private  ExporterChangeableWrapper doLocalExport(final Invoker originInvoker, URL providerUrl) {
    String key = getCacheKey(originInvoker);

    // 加入緩存
    return (ExporterChangeableWrapper) bounds.computeIfAbsent(key, s -> {
        // 創(chuàng)建 Invoker 為委托類對象
        Invoker invokerDelegate = new InvokerDelegate<>(originInvoker, providerUrl);
        // 調用 protocol 的 export 方法暴露服務
        return new ExporterChangeableWrapper<>((Exporter) protocol.export(invokerDelegate), originInvoker);
    });
}

這里的邏輯比較簡單,主要是在這里根據(jù)不同的協(xié)議配置,調用不同的protocol實現(xiàn)。跟暴露到本地的時候實現(xiàn)InjvmProtocol一樣。我這里假設配置選用的是dubbo協(xié)議,來繼續(xù)下面的介紹。

DubboProtocol的export()

《dubbo源碼解析(二十四)遠程調用——dubbo協(xié)議》中的(三)DubboProtocol有export()相關的源碼分析, 從源碼中可以看出做了一些本地存根的處理,關鍵的就是openServer,來啟動服務器。

DubboProtocol的openServer()

《dubbo源碼解析(二十四)遠程調用——dubbo協(xié)議》中的(三)DubboProtocol有openServer()相關的源碼分析, 不過該文章中的代碼是2.6.x的代碼,最新的版本中加入了 DCL。其中reset方法則是重置服務器的一些配置。例如在同一臺機器上(單網(wǎng)卡),同一個端口上僅允許啟動一個服務器實例。若某個端口上已有服務器實例,此時則調用 reset 方法重置服務器的一些配置。主要來看其中的createServer()方法。

DubboProtocol的createServer()

《dubbo源碼解析(二十四)遠程調用——dubbo協(xié)議》中的(三)DubboProtocol有createServer()相關的源碼分析,其中最新版本的默認遠程通訊服務端實現(xiàn)方式已經(jīng)改為netty4。該方法大致可以分為以下幾個步驟:

對服務端遠程通訊服務端實現(xiàn)方式配置是否支持的檢測

創(chuàng)建服務器實例,也就是調用bind()方法

對服務端遠程通訊客戶端實現(xiàn)方式配置是否支持的檢測

Exchangers的bind()

可以參考《dubbo源碼解析(十)遠程通信——Exchange層》中的(二十一)Exchangers。

public static ExchangeServer bind(URL url, ExchangeHandler handler) throws RemotingException {
    if (url == null) {
        throw new IllegalArgumentException("url == null");
    }
    if (handler == null) {
        throw new IllegalArgumentException("handler == null");
    }
    url = url.addParameterIfAbsent(Constants.CODEC_KEY, "exchange");
    // 獲取 Exchanger,默認為 HeaderExchanger。
    // 緊接著調用 HeaderExchanger 的 bind 方法創(chuàng)建 ExchangeServer 實例
    return getExchanger(url).bind(url, handler);
}
HeaderExchanger的bind()

可以參考《dubbo源碼解析(十)遠程通信——Exchange層》(十六)HeaderExchanger,其中bind()方法做了大致以下步驟:

創(chuàng)建HeaderExchangeHandler

創(chuàng)建DecodeHandler

Transporters.bind(),創(chuàng)建服務器實例。

創(chuàng)建HeaderExchangeServer

其中HeaderExchangeHandler、DecodeHandler、HeaderExchangeServer可以參考《dubbo源碼解析(十)遠程通信——Exchange層》中的講解。

Transporters的bind()(6)
public static Server bind(URL url, ChannelHandler... handlers) throws RemotingException {
    if (url == null) {
        throw new IllegalArgumentException("url == null");
    }
    if (handlers == null || handlers.length == 0) {
        throw new IllegalArgumentException("handlers == null");
    }
    ChannelHandler handler;
    if (handlers.length == 1) {
        handler = handlers[0];
    } else {
        // 如果 handlers 元素數(shù)量大于1,則創(chuàng)建 ChannelHandler 分發(fā)器
        handler = new ChannelHandlerDispatcher(handlers);
    }
    // 獲取自適應 Transporter 實例,并調用實例方法
    return getTransporter().bind(url, handler);
}

getTransporter() 方法獲取的 Transporter 是在運行時動態(tài)創(chuàng)建的,類名為 TransporterAdaptive,也就是自適應拓展類。TransporterAdaptive 會在運行時根據(jù)傳入的 URL 參數(shù)決定加載什么類型的 Transporter,默認為基于Netty4的實現(xiàn)。假設是 NettyTransporter 的 bind 方法。

NettyTransporter的bind()(6)

可以參考《dubbo源碼解析(十七)遠程通信——Netty4》的(六)NettyTransporter。

public Server bind(URL url, ChannelHandler listener) throws RemotingException {
    // 創(chuàng)建NettyServer
    return new NettyServer(url, listener);
}
NettyServer的構造方法(7)

可以參考《dubbo源碼解析(十七)遠程通信——Netty4》的(五)NettyServer

public NettyServer(URL url, ChannelHandler handler) throws RemotingException {
    super(url, ChannelHandlers.wrap(handler, ExecutorUtil.setThreadName(url, SERVER_THREAD_POOL_NAME)));
}

調用的是父類AbstractServer構造方法

AbstractServer的構造方法(7)

可以參考《dubbo源碼解析(九)遠程通信——Transport層》的(三)AbstractServer中的構造方法。

服務器實例創(chuàng)建完以后,就是開啟服務器了,AbstractServer中的doOpen是抽象方法,還是拿netty4來講解,也就是看NettyServer的doOpen()的方法。

NettyServer的doOpen()

可以參考《dubbo源碼解析(十七)遠程通信——Netty4》中的(五)NettyServer中的源碼分析。這里執(zhí)行完成后,服務器被開啟,服務也暴露出來了。接下來就是講解服務注冊的內容。

服務注冊(9)

dubbo服務注冊并不是必須的,因為dubbo支持直連的方式就可以繞過注冊中心。直連的方式很多時候用來做服務測試。

回過頭去看一下RegistryProtocol的 export()方法的分割線下面部分。其中服務注冊先調用的是register()方法。

RegistryProtocol的register()
public void register(URL registryUrl, URL registeredProviderUrl) {
    // 獲取 Registry
    Registry registry = registryFactory.getRegistry(registryUrl);
    // 注冊服務
    registry.register(registeredProviderUrl);
}

所以服務注冊大致可以分為兩步:

獲得注冊中心實例

注冊服務

獲得注冊中心首先執(zhí)行的是AbstractRegistryFactory的getRegistry()方法

AbstractRegistryFactory的getRegistry()

可以參考《dubbo源碼解析(三)注冊中心——開篇》的(七)support包下的AbstractRegistryFactory中的源碼解析。大概的邏輯就是先從緩存中取,如果沒有命中,則創(chuàng)建注冊中心實例,這里的createRegistry()是一個抽象方法,具體的實現(xiàn)邏輯由子類完成,假設這里使用zookeeper作為注冊中心,則調用的是ZookeeperRegistryFactory的createRegistry()。

ZookeeperRegistryFactory的createRegistry()
public Registry createRegistry(URL url) {
    return new ZookeeperRegistry(url, zookeeperTransporter);
}

就是創(chuàng)建了一個ZookeeperRegistry,執(zhí)行了ZookeeperRegistry的構造方法。

ZookeeperRegistry的構造方法

可以參考《dubbo源碼解析(七)注冊中心——zookeeper》的(一)ZookeeperRegistry中的源碼分析。大致的邏輯可以分為以下幾個步驟:

創(chuàng)建zookeeper客戶端

添加監(jiān)聽器

主要看ZookeeperTransporter的connect方法,因為當connect方法執(zhí)行完后,注冊中心創(chuàng)建過程就結束了。首先執(zhí)行的是AbstractZookeeperTransporter的connect方法。

AbstractZookeeperTransporter的connect()
public ZookeeperClient connect(URL url) {
    ZookeeperClient zookeeperClient;
    // 獲得所有url地址
    List addressList = getURLBackupAddress(url);
    // The field define the zookeeper server , including protocol, host, port, username, password
    // 從緩存中查找可用的客戶端,如果有,則直接返回
    if ((zookeeperClient = fetchAndUpdateZookeeperClientCache(addressList)) != null && zookeeperClient.isConnected()) {
        logger.info("find valid zookeeper client from the cache for address: " + url);
        return zookeeperClient;
    }
    // avoid creating too many connections, so add lock
    synchronized (zookeeperClientMap) {
        if ((zookeeperClient = fetchAndUpdateZookeeperClientCache(addressList)) != null && zookeeperClient.isConnected()) {
            logger.info("find valid zookeeper client from the cache for address: " + url);
            return zookeeperClient;
        }

        // 創(chuàng)建客戶端
        zookeeperClient = createZookeeperClient(toClientURL(url));
        logger.info("No valid zookeeper client found from cache, therefore create a new client for url. " + url);
        // 加入緩存
        writeToClientMap(addressList, zookeeperClient);
    }
    return zookeeperClient;
}

看上面的源碼,主要是執(zhí)行了createZookeeperClient()方法,而該方法是一個抽象方法,由子類實現(xiàn),這里是CuratorZookeeperTransporter的createZookeeperClient()

CuratorZookeeperTransporter的createZookeeperClient()
public ZookeeperClient createZookeeperClient(URL url) {
    return new CuratorZookeeperClient(url);
}

這里就是執(zhí)行了CuratorZookeeperClient的構造方法。

CuratorZookeeperClient的構造方法

可以參考《dubbo源碼解析(十八)遠程通信——Zookeeper》的(四)CuratorZookeeperClient中的源碼分析,其中邏輯主要用于創(chuàng)建和啟動 CuratorFramework 實例,基本都是調用Curator框架的API。

創(chuàng)建完注冊中心的實例后,我們就要進行注冊服務了。也就是調用的是FailbackRegistry的register()方法。

FailbackRegistry的register()

可以參考《dubbo源碼解析(三)注冊中心——開篇》的(六)support包下的FailbackRegistry中的源碼分析。可以看到關鍵是執(zhí)行了doRegister()方法,該方法是抽象方法,由子類完成。這里因為假設是zookeeper,所以執(zhí)行的是ZookeeperRegistry的doRegister()。

ZookeeperRegistry的doRegister()

可以參考《dubbo源碼解析(七)注冊中心——zookeeper》的(一)ZookeeperRegistry中的源代碼,可以看到邏輯就是調用Zookeeper 客戶端創(chuàng)建服務節(jié)點。節(jié)點路徑由 toUrlPath 方法生成。而這里create方法執(zhí)行的是AbstractZookeeperClient的create() 方法

AbstractZookeeperClient的create()

可以參考dubbo源碼解析(十八)遠程通信——Zookeeper》的(二)AbstractZookeeperClient中的源代碼分析。createEphemeral()和createPersistent()是抽象方法,具體實現(xiàn)由子類完成,也就是CuratorZookeeperClient類。代碼邏輯比較簡單。我就不再贅述。到這里為止,服務也就注冊完成。

關于向注冊中心進行訂閱 override 數(shù)據(jù)的規(guī)則在最新版本有一些大變動,跟2.6.x及以前的都不一樣。所以這部分內容在新特性中去講解。

后記
參考官方文檔:https://dubbo.apache.org/zh-c...

該文章講解了dubbo的服務暴露過程,也是為了之后講2.7新特性做鋪墊,下一篇講解服務的引用過程。

文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉載請注明本文地址:http://systransis.cn/yun/74258.html

相關文章

  • dubbo源碼解析四十八)異步化改造

    摘要:大揭秘異步化改造目標從源碼的角度分析的新特性中對于異步化的改造原理??丛创a解析四十六消費端發(fā)送請求過程講到的十四的,在以前的邏輯會直接在方法中根據(jù)配置區(qū)分同步異步單向調用。改為關于可以參考源碼解析十遠程通信層的六。 2.7大揭秘——異步化改造 目標:從源碼的角度分析2.7的新特性中對于異步化的改造原理。 前言 dubbo中提供了很多類型的協(xié)議,關于協(xié)議的系列可以查看下面的文章: du...

    lijinke666 評論0 收藏0
  • dubbo源碼解析四十六)消費端發(fā)送請求過程

    摘要:可以參考源碼解析二十四遠程調用協(xié)議的八。十六的該類也是用了適配器模式,該類主要的作用就是增加了心跳功能,可以參考源碼解析十遠程通信層的四。二十的可以參考源碼解析十七遠程通信的一。 2.7大揭秘——消費端發(fā)送請求過程 目標:從源碼的角度分析一個服務方法調用經(jīng)歷怎么樣的磨難以后到達服務端。 前言 前一篇文章講到的是引用服務的過程,引用服務無非就是創(chuàng)建出一個代理。供消費者調用服務的相關方法。...

    fish 評論0 收藏0
  • dubbo源碼解析四十七)服務端處理請求過程

    摘要:而存在的意義就是保證請求或響應對象可在線程池中被解碼,解碼完成后,就會分發(fā)到的。 2.7大揭秘——服務端處理請求過程 目標:從源碼的角度分析服務端接收到請求后的一系列操作,最終把客戶端需要的值返回。 前言 上一篇講到了消費端發(fā)送請求的過程,該篇就要將服務端處理請求的過程。也就是當服務端收到請求數(shù)據(jù)包后的一系列處理以及如何返回最終結果。我們也知道消費端在發(fā)送請求的時候已經(jīng)做了編碼,所以我...

    yzzz 評論0 收藏0
  • dubbo源碼解析四十五)服務引用過程

    摘要:服務引用過程目標從源碼的角度分析服務引用過程。并保留服務提供者的部分配置,比如版本,,時間戳等最后將合并后的配置設置為查詢字符串中。的可以參考源碼解析二十三遠程調用的一的源碼分析。 dubbo服務引用過程 目標:從源碼的角度分析服務引用過程。 前言 前面服務暴露過程的文章講解到,服務引用有兩種方式,一種就是直連,也就是直接指定服務的地址來進行引用,這種方式更多的時候被用來做服務測試,不...

    xiaowugui666 評論0 收藏0
  • dubbo源碼解析四十三)2.7新特性

    摘要:大揭秘目標了解的新特性,以及版本升級的引導。四元數(shù)據(jù)改造我們知道以前的版本只有注冊中心,注冊中心的有數(shù)十個的鍵值對,包含了一個服務所有的元數(shù)據(jù)。 DUBBO——2.7大揭秘 目標:了解2.7的新特性,以及版本升級的引導。 前言 我們知道Dubbo在2011年開源,停止更新了一段時間。在2017 年 9 月 7 日,Dubbo 悄悄的在 GitHub 發(fā)布了 2.5.4 版本。隨后,版本...

    qqlcbb 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<