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

資訊專欄INFORMATION COLUMN

dubbo源碼解析(三十七)集群——directory

blastz / 840人閱讀

摘要:首先將根據(jù)路由規(guī)則服務(wù)提供者和配置規(guī)則三種類型分開,分別放入三個(gè)集合,然后對每個(gè)集合進(jìn)行修改或者通知設(shè)置禁止訪問置空關(guān)閉所有的關(guān)閉禁止訪問引用老的傳入的為空,說明是路由規(guī)則或配置規(guī)則發(fā)生改變,此時(shí)是空的,直接使用。

集群——directory
目標(biāo):介紹dubbo中集群的目錄,介紹dubbo-cluster下directory包的源碼。
前言

我在前面的文章中也提到了Directory可以看成是多個(gè)Invoker的集合,Directory 的用途是保存 Invoker,其實(shí)現(xiàn)類 RegistryDirectory 是一個(gè)動態(tài)服務(wù)目錄,可感知注冊中心配置的變化,它所持有的 Inovker 列表會隨著注冊中心內(nèi)容的變化而變化。每次變化后,RegistryDirectory 會動態(tài)增刪 Inovker,那在之前文章中我忽略了RegistryDirectory的源碼分析,在本文中來補(bǔ)充。

源碼分析 (一)AbstractDirectory

該類實(shí)現(xiàn)了Directory接口,

1.屬性
// logger
private static final Logger logger = LoggerFactory.getLogger(AbstractDirectory.class);

/**
 * url對象
 */
private final URL url;

/**
 * 是否銷毀
 */
private volatile boolean destroyed = false;

/**
 * 消費(fèi)者端url
 */
private volatile URL consumerUrl;

/**
 * 路由集合
 */
private volatile List routers;
2.list
@Override
public List> list(Invocation invocation) throws RpcException {
    // 如果銷毀,則拋出異常
    if (destroyed) {
        throw new RpcException("Directory already destroyed .url: " + getUrl());
    }
    // 調(diào)用doList來獲得Invoker集合
    List> invokers = doList(invocation);
    // 獲得路由集合
    List localRouters = this.routers; // local reference
    if (localRouters != null && !localRouters.isEmpty()) {
        // 遍歷路由
        for (Router router : localRouters) {
            try {
                if (router.getUrl() == null || router.getUrl().getParameter(Constants.RUNTIME_KEY, false)) {
                    // 根據(jù)路由規(guī)則選擇符合規(guī)則的invoker集合
                    invokers = router.route(invokers, getConsumerUrl(), invocation);
                }
            } catch (Throwable t) {
                logger.error("Failed to execute router: " + getUrl() + ", cause: " + t.getMessage(), t);
            }
        }
    }
    return invokers;
}

該方法是生成invoker集合的邏輯實(shí)現(xiàn)。其中doList是抽象方法,交由子類來實(shí)現(xiàn)。

3.setRouters
protected void setRouters(List routers) {
    // copy list
    // 復(fù)制路由集合
    routers = routers == null ? new ArrayList() : new ArrayList(routers);
    // append url router
    // 獲得路由的配置
    String routerkey = url.getParameter(Constants.ROUTER_KEY);
    if (routerkey != null && routerkey.length() > 0) {
        // 加載路由工廠
        RouterFactory routerFactory = ExtensionLoader.getExtensionLoader(RouterFactory.class).getExtension(routerkey);
        // 加入集合
        routers.add(routerFactory.getRouter(url));
    }
    // append mock invoker selector
    // 加入服務(wù)降級路由
    routers.add(new MockInvokersSelector());
    // 排序
    Collections.sort(routers);
    this.routers = routers;
}
(二)StaticDirectory

靜態(tài) Directory 實(shí)現(xiàn)類,將傳入的 invokers 集合,封裝成靜態(tài)的 Directory 對象。

public class StaticDirectory extends AbstractDirectory {

    private final List> invokers;

    public StaticDirectory(List> invokers) {
        this(null, invokers, null);
    }

    public StaticDirectory(List> invokers, List routers) {
        this(null, invokers, routers);
    }

    public StaticDirectory(URL url, List> invokers) {
        this(url, invokers, null);
    }

    public StaticDirectory(URL url, List> invokers, List routers) {
        super(url == null && invokers != null && !invokers.isEmpty() ? invokers.get(0).getUrl() : url, routers);
        if (invokers == null || invokers.isEmpty())
            throw new IllegalArgumentException("invokers == null");
        this.invokers = invokers;
    }

    @Override
    public Class getInterface() {
        return invokers.get(0).getInterface();
    }

    @Override
    public boolean isAvailable() {
        if (isDestroyed()) {
            return false;
        }
        // 遍歷invokers,如果有一個(gè)可用,則可用
        for (Invoker invoker : invokers) {
            if (invoker.isAvailable()) {
                return true;
            }
        }
        return false;
    }

    @Override
    public void destroy() {
        if (isDestroyed()) {
            return;
        }
        super.destroy();
        // 遍歷invokers,銷毀所有的invoker
        for (Invoker invoker : invokers) {
            invoker.destroy();
        }
        // 清除集合
        invokers.clear();
    }

    @Override
    protected List> doList(Invocation invocation) throws RpcException {

        return invokers;
    }

}

該類我就不多講解,比較簡單。

(三)RegistryDirectory

該類繼承了AbstractDirectory類,是基于注冊中心的動態(tài) Directory 實(shí)現(xiàn)類,會根據(jù)注冊中心的推送變更 List

1.屬性
private static final Logger logger = LoggerFactory.getLogger(RegistryDirectory.class);

/**
 * cluster實(shí)現(xiàn)類對象
 */
private static final Cluster cluster = ExtensionLoader.getExtensionLoader(Cluster.class).getAdaptiveExtension();

/**
 * 路由工廠
 */
private static final RouterFactory routerFactory = ExtensionLoader.getExtensionLoader(RouterFactory.class).getAdaptiveExtension();

/**
 * 配置規(guī)則工廠
 */
private static final ConfiguratorFactory configuratorFactory = ExtensionLoader.getExtensionLoader(ConfiguratorFactory.class).getAdaptiveExtension();

/**
 * 服務(wù)key
 */
private final String serviceKey; // Initialization at construction time, assertion not null
/**
 * 服務(wù)類型
 */
private final Class serviceType; // Initialization at construction time, assertion not null
/**
 * 消費(fèi)者URL的配置項(xiàng) Map
 */
private final Map queryMap; // Initialization at construction time, assertion not null
/**
 * 原始的目錄 URL
 */
private final URL directoryUrl; // Initialization at construction time, assertion not null, and always assign non null value
/**
 * 服務(wù)方法集合
 */
private final String[] serviceMethods;
/**
 * 是否使用多分組
 */
private final boolean multiGroup;
/**
 * 協(xié)議
 */
private Protocol protocol; // Initialization at the time of injection, the assertion is not null
/**
 * 注冊中心
 */
private Registry registry; // Initialization at the time of injection, the assertion is not null
/**
 *  是否禁止訪問
 */
private volatile boolean forbidden = false;

/**
 * 覆蓋目錄的url
 */
private volatile URL overrideDirectoryUrl; // Initialization at construction time, assertion not null, and always assign non null value

/**
 * override rules
 * Priority: override>-D>consumer>provider
 * Rule one: for a certain provider 
 * Rule two: for all providers <* ,timeout=5000>
 * 配置規(guī)則數(shù)組
 */
private volatile List configurators; // The initial value is null and the midway may be assigned to null, please use the local variable reference

// Map cache service url to invoker mapping.
/**
 * url與服務(wù)提供者 Invoker 集合的映射緩存
 */
private volatile Map> urlInvokerMap; // The initial value is null and the midway may be assigned to null, please use the local variable reference

// Map cache service method to invokers mapping.
/**
 * 方法名和服務(wù)提供者 Invoker 集合的映射緩存
 */
private volatile Map>> methodInvokerMap; // The initial value is null and the midway may be assigned to null, please use the local variable reference

// Set cache invokeUrls to invokers mapping.
/**
 * 服務(wù)提供者Invoker 集合緩存
 */
private volatile Set cachedInvokerUrls; // The initial value is null and the midway may be assigned to null, please use the local variable reference
2.toConfigurators
public static List toConfigurators(List urls) {
    // 如果為空,則返回空集合
    if (urls == null || urls.isEmpty()) {
        return Collections.emptyList();
    }

    List configurators = new ArrayList(urls.size());
    // 遍歷url集合
    for (URL url : urls) {
        //如果是協(xié)議是empty的值,則清空配置集合
        if (Constants.EMPTY_PROTOCOL.equals(url.getProtocol())) {
            configurators.clear();
            break;
        }
        // 覆蓋的參數(shù)集合
        Map override = new HashMap(url.getParameters());
        //The anyhost parameter of override may be added automatically, it can"t change the judgement of changing url
        // 覆蓋的anyhost參數(shù)可以自動添加,也不能改變更改url的判斷
        override.remove(Constants.ANYHOST_KEY);
        // 如果需要覆蓋添加的值為0,則清空配置
        if (override.size() == 0) {
            configurators.clear();
            continue;
        }
        // 加入配置規(guī)則集合
        configurators.add(configuratorFactory.getConfigurator(url));
    }
    // 排序
    Collections.sort(configurators);
    return configurators;
}

該方法是處理配置規(guī)則url集合,轉(zhuǎn)換覆蓋url映射以便在重新引用時(shí)使用,每次發(fā)送所有規(guī)則,網(wǎng)址將被重新組裝和計(jì)算。

3.destroy
@Override
public void destroy() {
    // 如果銷毀了,則返回
    if (isDestroyed()) {
        return;
    }
    // unsubscribe.
    try {
        if (getConsumerUrl() != null && registry != null && registry.isAvailable()) {
            // 取消訂閱
            registry.unsubscribe(getConsumerUrl(), this);
        }
    } catch (Throwable t) {
        logger.warn("unexpeced error when unsubscribe service " + serviceKey + "from registry" + registry.getUrl(), t);
    }
    super.destroy(); // must be executed after unsubscribing
    try {
        // 清空所有的invoker
        destroyAllInvokers();
    } catch (Throwable t) {
        logger.warn("Failed to destroy service " + serviceKey, t);
    }
}

該方法是銷毀方法。

4.destroyAllInvokers
private void destroyAllInvokers() {
    Map> localUrlInvokerMap = this.urlInvokerMap; // local reference
    // 如果invoker集合不為空
    if (localUrlInvokerMap != null) {
        // 遍歷
        for (Invoker invoker : new ArrayList>(localUrlInvokerMap.values())) {
            try {
                // 銷毀invoker
                invoker.destroy();
            } catch (Throwable t) {
                logger.warn("Failed to destroy service " + serviceKey + " to provider " + invoker.getUrl(), t);
            }
        }
        // 清空集合
        localUrlInvokerMap.clear();
    }
    methodInvokerMap = null;
}

該方法是關(guān)閉所有的invoker服務(wù)。

5.notify
@Override
public synchronized void notify(List urls) {
    List invokerUrls = new ArrayList();
    List routerUrls = new ArrayList();
    List configuratorUrls = new ArrayList();
    // 遍歷url
    for (URL url : urls) {
        // 獲得協(xié)議
        String protocol = url.getProtocol();
        // 獲得類別
        String category = url.getParameter(Constants.CATEGORY_KEY, Constants.DEFAULT_CATEGORY);
        // 如果是路由規(guī)則
        if (Constants.ROUTERS_CATEGORY.equals(category)
                || Constants.ROUTE_PROTOCOL.equals(protocol)) {
            // 則在路由規(guī)則集合中加入
            routerUrls.add(url);
        } else if (Constants.CONFIGURATORS_CATEGORY.equals(category)
                || Constants.OVERRIDE_PROTOCOL.equals(protocol)) {
            // 如果是配置規(guī)則,則加入配置規(guī)則集合
            configuratorUrls.add(url);
        } else if (Constants.PROVIDERS_CATEGORY.equals(category)) {
            // 如果是服務(wù)提供者,則加入服務(wù)提供者集合
            invokerUrls.add(url);
        } else {
            logger.warn("Unsupported category " + category + " in notified url: " + url + " from registry " + getUrl().getAddress() + " to consumer " + NetUtils.getLocalHost());
        }
    }
    // configurators
    if (configuratorUrls != null && !configuratorUrls.isEmpty()) {
        // 處理配置規(guī)則url集合
        this.configurators = toConfigurators(configuratorUrls);
    }
    // routers
    if (routerUrls != null && !routerUrls.isEmpty()) {
        // 處理路由規(guī)則 URL 集合
        List routers = toRouters(routerUrls);
        if (routers != null) { // null - do nothing
            // 并且設(shè)置路由集合
            setRouters(routers);
        }
    }
    List localConfigurators = this.configurators; // local reference
    // merge override parameters
    this.overrideDirectoryUrl = directoryUrl;
    if (localConfigurators != null && !localConfigurators.isEmpty()) {
        // 遍歷配置規(guī)則集合 逐個(gè)進(jìn)行配置
        for (Configurator configurator : localConfigurators) {
            this.overrideDirectoryUrl = configurator.configure(overrideDirectoryUrl);
        }
    }
    // providers
    // 處理服務(wù)提供者 URL 集合
    refreshInvoker(invokerUrls);
}

當(dāng)服務(wù)有變化的時(shí)候,執(zhí)行該方法。首先將url根據(jù)路由規(guī)則、服務(wù)提供者和配置規(guī)則三種類型分開,分別放入三個(gè)集合,然后對每個(gè)集合進(jìn)行修改或者通知

6.refreshInvoker
private void refreshInvoker(List invokerUrls) {
    if (invokerUrls != null && invokerUrls.size() == 1 && invokerUrls.get(0) != null
            && Constants.EMPTY_PROTOCOL.equals(invokerUrls.get(0).getProtocol())) {
        // 設(shè)置禁止訪問
        this.forbidden = true; // Forbid to access
        // methodInvokerMap 置空
        this.methodInvokerMap = null; // Set the method invoker map to null
        // 關(guān)閉所有的invoker
        destroyAllInvokers(); // Close all invokers
    } else {
        // 關(guān)閉禁止訪問
        this.forbidden = false; // Allow to access
        // 引用老的 urlInvokerMap
        Map> oldUrlInvokerMap = this.urlInvokerMap; // local reference
        // 傳入的 invokerUrls 為空,說明是路由規(guī)則或配置規(guī)則發(fā)生改變,此時(shí) invokerUrls 是空的,直接使用 cachedInvokerUrls 。
        if (invokerUrls.isEmpty() && this.cachedInvokerUrls != null) {
            invokerUrls.addAll(this.cachedInvokerUrls);
        } else {
            // 否則把所有的invokerUrls加入緩存
            this.cachedInvokerUrls = new HashSet();
            this.cachedInvokerUrls.addAll(invokerUrls);//Cached invoker urls, convenient for comparison
        }
        // 如果invokerUrls為空,則直接返回
        if (invokerUrls.isEmpty()) {
            return;
        }
        // 將傳入的 invokerUrls ,轉(zhuǎn)成新的 urlInvokerMap
        Map> newUrlInvokerMap = toInvokers(invokerUrls);// Translate url list to Invoker map
        // 轉(zhuǎn)換出新的 methodInvokerMap
        Map>> newMethodInvokerMap = toMethodInvokers(newUrlInvokerMap); // Change method name to map Invoker Map
        // state change
        // If the calculation is wrong, it is not processed.
        // 如果為空,則打印錯(cuò)誤日志并且返回
        if (newUrlInvokerMap == null || newUrlInvokerMap.size() == 0) {
            logger.error(new IllegalStateException("urls to invokers error .invokerUrls.size :" + invokerUrls.size() + ", invoker.size :0. urls :" + invokerUrls.toString()));
            return;
        }
        // 若服務(wù)引用多 group ,則按照 method + group 聚合 Invoker 集合
        this.methodInvokerMap = multiGroup ? toMergeMethodInvokerMap(newMethodInvokerMap) : newMethodInvokerMap;
        this.urlInvokerMap = newUrlInvokerMap;
        try {
            // 銷毀不再使用的 Invoker 集合
            destroyUnusedInvokers(oldUrlInvokerMap, newUrlInvokerMap); // Close the unused Invoker
        } catch (Exception e) {
            logger.warn("destroyUnusedInvokers error. ", e);
        }
    }
}

該方法是處理服務(wù)提供者 URL 集合。根據(jù) invokerURL 列表轉(zhuǎn)換為 invoker 列表。轉(zhuǎn)換規(guī)則如下:

如果 url 已經(jīng)被轉(zhuǎn)換為 invoker ,則不在重新引用,直接從緩存中獲取,注意如果 url 中任何一個(gè)參數(shù)變更也會重新引用。

如果傳入的 invoker 列表不為空,則表示最新的 invoker 列表。

如果傳入的 invokerUrl 列表是空,則表示只是下發(fā)的 override 規(guī)則或 route 規(guī)則,需要重新交叉對比,決定是否需要重新引用。

7.toMergeMethodInvokerMap
private Map>> toMergeMethodInvokerMap(Map>> methodMap) {
    // 循環(huán)方法,按照 method + group 聚合 Invoker 集合
    Map>> result = new HashMap>>();
    // 遍歷方法集合
    for (Map.Entry>> entry : methodMap.entrySet()) {
        // 獲得方法
        String method = entry.getKey();
        // 獲得invoker集合
        List> invokers = entry.getValue();
        // 獲得組集合
        Map>> groupMap = new HashMap>>();
        // 遍歷invoker集合
        for (Invoker invoker : invokers) {
            // 獲得url攜帶的組配置
            String group = invoker.getUrl().getParameter(Constants.GROUP_KEY, "");
            // 獲得該組對應(yīng)的invoker集合
            List> groupInvokers = groupMap.get(group);
            // 如果為空,則新創(chuàng)建一個(gè),然后加入集合
            if (groupInvokers == null) {
                groupInvokers = new ArrayList>();
                groupMap.put(group, groupInvokers);
            }
            groupInvokers.add(invoker);
        }
        // 如果只有一個(gè)組
        if (groupMap.size() == 1) {
            // 返回該組的invoker集合
            result.put(method, groupMap.values().iterator().next());
        } else if (groupMap.size() > 1) {
            // 如果不止一個(gè)組
            List> groupInvokers = new ArrayList>();
            // 遍歷組
            for (List> groupList : groupMap.values()) {
                // 每次從集群中選擇一個(gè)invoker加入groupInvokers
                groupInvokers.add(cluster.join(new StaticDirectory(groupList)));
            }
            // 加入需要返回的集合
            result.put(method, groupInvokers);
        } else {
            result.put(method, invokers);
        }
    }
    return result;
}

該方法是通過按照 method + group 來聚合 Invoker 集合。

8.toRouters
private List toRouters(List urls) {
    List routers = new ArrayList();
    // 如果為空,則直接返回空集合
    if (urls == null || urls.isEmpty()) {
        return routers;
    }
    if (urls != null && !urls.isEmpty()) {
        // 遍歷url集合
        for (URL url : urls) {
            // 如果為empty協(xié)議,則直接跳過
            if (Constants.EMPTY_PROTOCOL.equals(url.getProtocol())) {
                continue;
            }
            // 獲得路由規(guī)則
            String routerType = url.getParameter(Constants.ROUTER_KEY);
            if (routerType != null && routerType.length() > 0) {
                // 設(shè)置協(xié)議
                url = url.setProtocol(routerType);
            }
            try {
                // 獲得路由
                Router router = routerFactory.getRouter(url);
                if (!routers.contains(router))
                    // 加入集合
                    routers.add(router);
            } catch (Throwable t) {
                logger.error("convert router url to router error, url: " + url, t);
            }
        }
    }
    return routers;
}

該方法是對url集合進(jìn)行路由的解析,返回路由集合。

9.toInvokers
private Map> toInvokers(List urls) {
    Map> newUrlInvokerMap = new HashMap>();
    // 如果為空,則返回空集合
    if (urls == null || urls.isEmpty()) {
        return newUrlInvokerMap;
    }
    Set keys = new HashSet();
    // 獲得引用服務(wù)的協(xié)議
    String queryProtocols = this.queryMap.get(Constants.PROTOCOL_KEY);
    // 遍歷url
    for (URL providerUrl : urls) {
        // If protocol is configured at the reference side, only the matching protocol is selected
        // 如果在參考側(cè)配置協(xié)議,則僅選擇匹配協(xié)議
        if (queryProtocols != null && queryProtocols.length() > 0) {
            boolean accept = false;
            // 分割協(xié)議
            String[] acceptProtocols = queryProtocols.split(",");
            // 遍歷協(xié)議
            for (String acceptProtocol : acceptProtocols) {
                // 如果匹配,則是接受的協(xié)議
                if (providerUrl.getProtocol().equals(acceptProtocol)) {
                    accept = true;
                    break;
                }
            }
            if (!accept) {
                continue;
            }
        }
        // 如果協(xié)議是empty,則跳過
        if (Constants.EMPTY_PROTOCOL.equals(providerUrl.getProtocol())) {
            continue;
        }
        // 如果該協(xié)議不是dubbo支持的,則打印錯(cuò)誤日志,跳過
        if (!ExtensionLoader.getExtensionLoader(Protocol.class).hasExtension(providerUrl.getProtocol())) {
            logger.error(new IllegalStateException("Unsupported protocol " + providerUrl.getProtocol() + " in notified url: " + providerUrl + " from registry " + getUrl().getAddress() + " to consumer " + NetUtils.getLocalHost()
                    + ", supported protocol: " + ExtensionLoader.getExtensionLoader(Protocol.class).getSupportedExtensions()));
            continue;
        }
        // 合并url參數(shù)
        URL url = mergeUrl(providerUrl);

        String key = url.toFullString(); // The parameter urls are sorted
        if (keys.contains(key)) { // Repeated url
            continue;
        }
        // 添加到keys
        keys.add(key);
        // Cache key is url that does not merge with consumer side parameters, regardless of how the consumer combines parameters, if the server url changes, then refer again
        // 如果服務(wù)端 URL 發(fā)生變化,則重新 refer 引用
        Map> localUrlInvokerMap = this.urlInvokerMap; // local reference
        Invoker invoker = localUrlInvokerMap == null ? null : localUrlInvokerMap.get(key);
        if (invoker == null) { // Not in the cache, refer again
            try {
                // 判斷是否開啟
                boolean enabled = true;
                // 獲得enabled配置
                if (url.hasParameter(Constants.DISABLED_KEY)) {
                    enabled = !url.getParameter(Constants.DISABLED_KEY, false);
                } else {
                    enabled = url.getParameter(Constants.ENABLED_KEY, true);
                }
                // 若開啟,創(chuàng)建 Invoker 對象
                if (enabled) {
                    invoker = new InvokerDelegate(protocol.refer(serviceType, url), url, providerUrl);
                }
            } catch (Throwable t) {
                logger.error("Failed to refer invoker for interface:" + serviceType + ",url:(" + url + ")" + t.getMessage(), t);
            }
            // 添加到 newUrlInvokerMap 中
            if (invoker != null) { // Put new invoker in cache
                newUrlInvokerMap.put(key, invoker);
            }
        } else {
            newUrlInvokerMap.put(key, invoker);
        }
    }
    // 清空 keys
    keys.clear();
    return newUrlInvokerMap;
}

該方法是將url轉(zhuǎn)換為調(diào)用者,如果url已被引用,則不會重新引用。

10.mergeUrl
private URL mergeUrl(URL providerUrl) {
    // 合并消費(fèi)端參數(shù)
    providerUrl = ClusterUtils.mergeUrl(providerUrl, queryMap); // Merge the consumer side parameters

    // 合并配置規(guī)則
    List localConfigurators = this.configurators; // local reference
    if (localConfigurators != null && !localConfigurators.isEmpty()) {
        for (Configurator configurator : localConfigurators) {
            providerUrl = configurator.configure(providerUrl);
        }
    }

    // 不檢查連接是否成功,總是創(chuàng)建 Invoker
    providerUrl = providerUrl.addParameter(Constants.CHECK_KEY, String.valueOf(false)); // Do not check whether the connection is successful or not, always create Invoker!

    // The combination of directoryUrl and override is at the end of notify, which can"t be handled here
    // 合并提供者參數(shù),因?yàn)?directoryUrl 與 override 合并是在 notify 的最后,這里不能夠處理
    this.overrideDirectoryUrl = this.overrideDirectoryUrl.addParametersIfAbsent(providerUrl.getParameters()); // Merge the provider side parameters

    // 1.0版本兼容
    if ((providerUrl.getPath() == null || providerUrl.getPath().length() == 0)
            && "dubbo".equals(providerUrl.getProtocol())) { // Compatible version 1.0
        //fix by tony.chenl DUBBO-44
        String path = directoryUrl.getParameter(Constants.INTERFACE_KEY);
        if (path != null) {
            int i = path.indexOf("/");
            if (i >= 0) {
                path = path.substring(i + 1);
            }
            i = path.lastIndexOf(":");
            if (i >= 0) {
                path = path.substring(0, i);
            }
            providerUrl = providerUrl.setPath(path);
        }
    }
    return providerUrl;
}

該方法是合并 URL 參數(shù),優(yōu)先級為配置規(guī)則 > 服務(wù)消費(fèi)者配置 > 服務(wù)提供者配置.

11.toMethodInvokers
private Map>> toMethodInvokers(Map> invokersMap) {
    Map>> newMethodInvokerMap = new HashMap>>();
    // According to the methods classification declared by the provider URL, the methods is compatible with the registry to execute the filtered methods
    List> invokersList = new ArrayList>();
    if (invokersMap != null && invokersMap.size() > 0) {
        // 遍歷調(diào)用者列表
        for (Invoker invoker : invokersMap.values()) {
            String parameter = invoker.getUrl().getParameter(Constants.METHODS_KEY);
            // 按服務(wù)提供者 URL 所聲明的 methods 分類
            if (parameter != null && parameter.length() > 0) {
                // 分割參數(shù)得到方法集合
                String[] methods = Constants.COMMA_SPLIT_PATTERN.split(parameter);
                if (methods != null && methods.length > 0) {
                    // 遍歷方法集合
                    for (String method : methods) {
                        if (method != null && method.length() > 0
                                && !Constants.ANY_VALUE.equals(method)) {
                            // 獲得該方法對應(yīng)的invoker,如果為空,則創(chuàng)建
                            List> methodInvokers = newMethodInvokerMap.get(method);
                            if (methodInvokers == null) {
                                methodInvokers = new ArrayList>();
                                newMethodInvokerMap.put(method, methodInvokers);
                            }
                            methodInvokers.add(invoker);
                        }
                    }
                }
            }
            invokersList.add(invoker);
        }
    }
    // 根據(jù)路由規(guī)則,匹配合適的 Invoker 集合。
    List> newInvokersList = route(invokersList, null);
    // 添加 `newInvokersList` 到 `newMethodInvokerMap` 中,表示該服務(wù)提供者的全量 Invoker 集合
    newMethodInvokerMap.put(Constants.ANY_VALUE, newInvokersList);
    if (serviceMethods != null && serviceMethods.length > 0) {
        // 循環(huán)方法,獲得每個(gè)方法路由匹配的invoker集合
        for (String method : serviceMethods) {
            List> methodInvokers = newMethodInvokerMap.get(method);
            if (methodInvokers == null || methodInvokers.isEmpty()) {
                methodInvokers = newInvokersList;
            }
            newMethodInvokerMap.put(method, route(methodInvokers, method));
        }
    }
    // sort and unmodifiable
    // 循環(huán)排序每個(gè)方法的 Invoker 集合,排序
    for (String method : new HashSet(newMethodInvokerMap.keySet())) {
        List> methodInvokers = newMethodInvokerMap.get(method);
        Collections.sort(methodInvokers, InvokerComparator.getComparator());
        newMethodInvokerMap.put(method, Collections.unmodifiableList(methodInvokers));
    }
    // 設(shè)置為不可變
    return Collections.unmodifiableMap(newMethodInvokerMap);
}

該方法是將調(diào)用者列表轉(zhuǎn)換為與方法的映射關(guān)系。

12.destroyUnusedInvokers
private void destroyUnusedInvokers(Map> oldUrlInvokerMap, Map> newUrlInvokerMap) {
    if (newUrlInvokerMap == null || newUrlInvokerMap.size() == 0) {
        destroyAllInvokers();
        return;
    }
    // check deleted invoker
    // 記錄已經(jīng)刪除的invoker
    List deleted = null;
    if (oldUrlInvokerMap != null) {
        Collection> newInvokers = newUrlInvokerMap.values();
        // 遍歷舊的invoker集合
        for (Map.Entry> entry : oldUrlInvokerMap.entrySet()) {
            if (!newInvokers.contains(entry.getValue())) {
                if (deleted == null) {
                    deleted = new ArrayList();
                }
                // 加入該invoker
                deleted.add(entry.getKey());
            }
        }
    }

    if (deleted != null) {
        // 遍歷需要?jiǎng)h除的invoker  url集合
        for (String url : deleted) {
            if (url != null) {
                // 移除該url
                Invoker invoker = oldUrlInvokerMap.remove(url);
                if (invoker != null) {
                    try {
                        // 銷毀invoker
                        invoker.destroy();
                        if (logger.isDebugEnabled()) {
                            logger.debug("destroy invoker[" + invoker.getUrl() + "] success. ");
                        }
                    } catch (Exception e) {
                        logger.warn("destroy invoker[" + invoker.getUrl() + "] faild. " + e.getMessage(), e);
                    }
                }
            }
        }
    }
}

該方法是銷毀不再使用的 Invoker 集合。

13.doList
@Override
public List> doList(Invocation invocation) {
    // 如果禁止訪問,則拋出異常
    if (forbidden) {
        // 1. No service provider 2. Service providers are disabled
        throw new RpcException(RpcException.FORBIDDEN_EXCEPTION,
            "No provider available from registry " + getUrl().getAddress() + " for service " + getConsumerUrl().getServiceKey() + " on consumer " +  NetUtils.getLocalHost()
                    + " use dubbo version " + Version.getVersion() + ", please check status of providers(disabled, not registered or in blacklist).");
    }
    List> invokers = null;
    Map>> localMethodInvokerMap = this.methodInvokerMap; // local reference
    if (localMethodInvokerMap != null && localMethodInvokerMap.size() > 0) {
        // 獲得方法名
        String methodName = RpcUtils.getMethodName(invocation);
        // 獲得參數(shù)名
        Object[] args = RpcUtils.getArguments(invocation);
        if (args != null && args.length > 0 && args[0] != null
                && (args[0] instanceof String || args[0].getClass().isEnum())) {
            // 根據(jù)第一個(gè)參數(shù)枚舉路由
            invokers = localMethodInvokerMap.get(methodName + "." + args[0]); // The routing can be enumerated according to the first parameter
        }
        if (invokers == null) {
            // 根據(jù)方法名獲得 Invoker 集合
            invokers = localMethodInvokerMap.get(methodName);
        }
        if (invokers == null) {
            // 使用全量 Invoker 集合。例如,`#$echo(name)` ,回聲方法
            invokers = localMethodInvokerMap.get(Constants.ANY_VALUE);
        }
        if (invokers == null) {
            // 使用 `methodInvokerMap` 第一個(gè) Invoker 集合。防御性編程。
            Iterator>> iterator = localMethodInvokerMap.values().iterator();
            if (iterator.hasNext()) {
                invokers = iterator.next();
            }
        }
    }
    return invokers == null ? new ArrayList>(0) : invokers;
}

該方法是通過會話域來獲得Invoker集合。

后記
該部分相關(guān)的源碼解析地址:https://github.com/CrazyHZM/i...

該文章講解了集群中關(guān)于directory實(shí)現(xiàn)的部分,關(guān)鍵是RegistryDirectory,其中涉及到眾多方法,需要好好品味。接下來我將開始對集群模塊關(guān)于loadbalance部分進(jìn)行講解。

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

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/77406.html

相關(guān)文章

  • dubbo源碼解析三十九)集群——merger

    摘要:源碼分析一創(chuàng)建該類實(shí)現(xiàn)了接口,是分組集合的集群實(shí)現(xiàn)。三工廠類,獲得指定類型的對象。后記該部分相關(guān)的源碼解析地址該文章講解了集群中關(guān)于分組聚合實(shí)現(xiàn)的部分。接下來我將開始對集群模塊關(guān)于路由部分進(jìn)行講解。 集群——merger 目標(biāo):介紹dubbo中集群的分組聚合,介紹dubbo-cluster下merger包的源碼。 前言 按組合并返回結(jié)果 ,比如菜單服務(wù),接口一樣,但有多種實(shí)現(xiàn),用gro...

    lscho 評論0 收藏0
  • dubbo源碼解析三十五)集群——cluster

    摘要:失敗安全,出現(xiàn)異常時(shí),直接忽略。失敗自動恢復(fù),在調(diào)用失敗后,返回一個(gè)空結(jié)果給服務(wù)提供者。源碼分析一該類實(shí)現(xiàn)了接口,是集群的抽象類。 集群——cluster 目標(biāo):介紹dubbo中集群容錯(cuò)的幾種模式,介紹dubbo-cluster下support包的源碼。 前言 集群容錯(cuò)還是很好理解的,就是當(dāng)你調(diào)用失敗的時(shí)候所作出的措施。先來看看有哪些模式: showImg(https://segmen...

    gself 評論0 收藏0
  • dubbo源碼解析(四十八)異步化改造

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

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

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

    fish 評論0 收藏0
  • dubbo源碼解析三十八)集群——LoadBalance

    摘要:集群目標(biāo)介紹中集群的負(fù)載均衡,介紹下包的源碼。源碼分析一該類實(shí)現(xiàn)了接口,是負(fù)載均衡的抽象類,提供了權(quán)重計(jì)算的功能。四該類是負(fù)載均衡基于一致性的邏輯實(shí)現(xiàn)。 集群——LoadBalance 目標(biāo):介紹dubbo中集群的負(fù)載均衡,介紹dubbo-cluster下loadBalance包的源碼。 前言 負(fù)載均衡,說的通俗點(diǎn)就是要一碗水端平。在這個(gè)時(shí)代,公平是很重要的,在網(wǎng)絡(luò)請求的時(shí)候同樣是這個(gè)...

    不知名網(wǎng)友 評論0 收藏0

發(fā)表評論

0條評論

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