//通知//当服务端有符合当前订阅的服务时被通知//主要目的:把提供服务的urls列表转换为invoker列表public synchronized void notify(Listurls) { ListinvokerUrls = new ArrayList(); ListrouterUrls = new ArrayList(); ListconfiguratorUrls = new ArrayList(); for (URL url : urls) { String protocol = url.getProtocol(); String category = url.getParameter(Constants.CATEGORY_KEY, Constants.DEFAULT_CATEGORY); if (Constants.ROUTERS_CATEGORY.equals(category) || Constants.ROUTE_PROTOCOL.equals(protocol)) { routerUrls.add(url); } else if (Constants.CONFIGURATORS_CATEGORY.equals(category) || Constants.OVERRIDE_PROTOCOL.equals(protocol)) { configuratorUrls.add(url); } else if (Constants.PROVIDERS_CATEGORY.equals(category)) { invokerUrls.add(url); } else { logger.warn("Unsupported category " + category + " in notified url: " + url + " from registry " + getUrl().getAddress() + " to consumer " + NetUtils.getLocalHost()); } } // configurators ... // routers ... // 合并override参数 // providers refreshInvoker(invokerUrls);}/** * 根据invokerURL列表转换为invoker列表。 */private void refreshInvoker(ListinvokerUrls){ // ... Map> newUrlInvokerMap = toInvokers(invokerUrls) ;// 将URL列表转成Invoker列表 Map>> newMethodInvokerMap = toMethodInvokers(newUrlInvokerMap); // 换方法名映射Invoker列表 // ... this.methodInvokerMap = multiGroup ? toMergeMethodInvokerMap(newMethodInvokerMap) : newMethodInvokerMap; this.urlInvokerMap = newUrlInvokerMap;}/** * 将urls转成invokers,如果url已经被refer过,不再重新引用。 * */private Map> toInvokers(Listurls) { Map> newUrlInvokerMap = new HashMap>(); for (URL providerUrl : urls) { URL url = mergeUrl(providerUrl); if (invoker == null) { // 缓存中没有,重新refer invoker = new InvokerDelegete(protocol.refer(serviceType, url), url, providerUrl);//调用protocol.refer newUrlInvokerMap.put(key, invoker); } }else { newUrlInvokerMap.put(key, invoker); } } return newUrlInvokerMap;}toInvokers将urls转成invokers。这里最终调用的还是protocol.refer,如DubboProtocol则将转成DubboInvoker,然后再作系列的包装-> DubboInvokerinvoker = new DubboInvoker(serviceType, url, getClients(url), invokers);//这里将ExchangeClient传给DubboInvoker 每个Invoker都维护了一个ExchangeClient的引用,并通过它和远程的Server进行通信。DubboProtocol:publicInvokerrefer(ClassserviceType, URL url) throws RpcException { // create rpc invoker. DubboInvokerinvoker = new DubboInvoker(serviceType, url, getClients(url), invokers);
invokers.add(invoker);
return invoker;
}
getClients(url) -> initClient(url) 建立客户端连接:
将requestHandler传递给HeaderExchangeHandler,并建立接收消息监听
-> getClients -> initClient -> client = Exchangers.connect(url ,requestHandler) -> HeaderExchangeHandler
-> HeaderExchangeHandler implements ChannelHandlerDelegate
received -> handleRequest -> handler.reply(channel, msg) -> invoker.invoke(inv) -> DubboInvoker.doInvoke
-> DubboInvoker.doInvoke -> currentClient.request //最终方法调用的时候会调用doInvoke,通过client发送请求
merge:
merge前 providerUrl =>
dubbo://130.130.10.146:20880/cn.tw.dubbo.intf.DemoService?anyhost=true&interface=cn.tw.dubbo.intf.DemoService&methods=sayHello&
application=hello-world-app&dubbo=2.5.3&pid=11812&side=provider×tamp=1498463368943
merge后 =>
dubbo://130.130.10.146:20880/cn.tw.dubbo.intf.DemoService?anyhost=true&interface=cn.tw.dubbo.intf.DemoService&methods=sayHello&
application=consumer-of-helloworld-app&check=false&dubbo=2.5.3&&pid=4604&side=consumer×tamp=1498724632185