接上篇博客《Dubbo源码深度解析(二)》继续讲。上篇博客主要讲的是Dubbo的SPI机制,以及Dubbo启动流程的初始化中的部分方法,本篇博客将继续讲剩下的方法,以及服务是如何发布出去的,先看看初始化方法中剩下的几个方法,如下:
data:image/s3,"s3://crabby-images/f65a8/f65a8f0f716645b5061d8a8c2e9f2dc53c2c48a4" alt=""
③ DubboBootstrap#loadRemoteConfigs()方法:
data:image/s3,"s3://crabby-images/0cb73/0cb7337d36bb584e0a7ed7275122ba2921b6b1f1" alt=""
data:image/s3,"s3://crabby-images/8ed2d/8ed2d1cc26869506d7672fbfbaf2c8c8c22a74b6" alt=""
data:image/s3,"s3://crabby-images/173f8/173f8c5c4725e6b508772918a73229bc77375b18" alt=""
这两个方法,就是从配置中心获取的配置中,找出前缀为:dubbo.registries. 和 dubbo.protocols. ,如供你没有配置,那请忽略,如果你配置了,当然可以获取到,并保存起来,保存逻辑为:
data:image/s3,"s3://crabby-images/f4c5c/f4c5c4b2aa2d1609f6fa891d56842787c4679808" alt=""
data:image/s3,"s3://crabby-images/24c95/24c95d9bff7f1c5d18055e67fc193c797e8436fd" alt=""
data:image/s3,"s3://crabby-images/5a24c/5a24c92b2ebc9d504034195d42700d19180a27e4" alt=""
data:image/s3,"s3://crabby-images/94cef/94cefed301976ccc289445cb02ae56f41c8a58dc" alt=""
可以知道,最终所有属性都是配置在ConfigManager的configsCache属性中。断点看看结果:
data:image/s3,"s3://crabby-images/9db78/9db78a83f78ab56eb31e1f5b0a101714cc4bebec" alt=""
但是断点中看到的结果,为什么key为registry和protocol的不止一个呢?因为我在配置中心的dubbo.properties做了修改,修改如下:
data:image/s3,"s3://crabby-images/82761/82761a3304eaacf577f56471d9a71172540ad106" alt=""
这是为了测试前面说的前缀为 dubbo.registries. 和 dubbo.protocols. 的情况,说白了就是配置多个注册中心和多个协议,这是Dubbo支持的。当然,仅仅在配置中心这样配置还不够,因为这里涉及到了hessian协议,需要在service-provider模块中引入相关的依赖,如下:
XML
<dependency>
<groupId>org.apache.dubbo</groupId>
<artifactId>dubbo-rpc-hessian</artifactId>
<version>2.7.10</version>
</dependency>
dubbo协议的依赖我在之前已经引入了,因此无需添加它的依赖。同时,由于在配置中心中灵配置了另一个nacos配置中心的地址,因此需要再开一个nacos服务,端口为:8849,Idea配置如下:
data:image/s3,"s3://crabby-images/27bb0/27bb0ad9f63da95e79fd28faeae825250d6ea7a4" alt=""
data:image/s3,"s3://crabby-images/3695b/3695b5fcb567621f867da8e0d9648eb1e25410c2" alt=""
重新启动服务提供方,发现在两个nacos服务上,在服务管理>服务列表,都可以查到,服务提供方的服务,先看看端口是8848的nacos服务,如下:
data:image/s3,"s3://crabby-images/86169/86169f30f7e175bb7c3a283b6909d5a69c6ffe0a" alt=""
查看详情,结果为:
data:image/s3,"s3://crabby-images/040d6/040d6177cb30d22e2227b5b33d1e42a809058c4d" alt=""
再看看端口是8849的nacos服务,如下:
data:image/s3,"s3://crabby-images/54bf1/54bf1aa3b67e9f02b3d94d507774589649274286" alt=""
查看详情,结果为:
data:image/s3,"s3://crabby-images/01a01/01a01cdcdda9355e4fe3ff746212ee4f80a5e8fe" alt=""
我发现,当前版本的Dubbo,如果使用hessian协议的话,服务无法使用netty4?看了下HessianProtocol,发现绑定服务的是这个属性,代码如下:
data:image/s3,"s3://crabby-images/6d914/6d91402d4d5b1ee1d411ce6e5365af54d784d36e" alt=""
只能使用Jetty、Tomcat或者Servlet,看看接口全限定名的文件,也是如此,如下:
data:image/s3,"s3://crabby-images/4307b/4307b6997612a697ef5b91d9cf3c440f9c289326" alt=""
那我们现在测一下,由于添加了hessian协议,因此在service-comsumer模块中也要加相关依赖,如下:
XML
<dependency>
<groupId>org.apache.dubbo</groupId>
<artifactId>dubbo-rpc-hessian</artifactId>
<version>2.7.10</version>
</dependency>
然后指定要使用的协议,再启动服务消费方,调用一个对外暴露的接口:http://localhost:9000/hello/getName/Tim,结果如下:
data:image/s3,"s3://crabby-images/eaa46/eaa46819fefa0a07c97bf7b4f103ac8c64203e49" alt=""
data:image/s3,"s3://crabby-images/35409/35409f2ebac05dec227a36e48c2fd8903b3ea468" alt=""
data:image/s3,"s3://crabby-images/8a2fb/8a2fb5db284048ab72943f97680ecf82151bd7f8" alt=""
如果使用dubbo协议呢?跟上面类似,如下:
data:image/s3,"s3://crabby-images/336a3/336a3549fa85e8836f9e47b19b0f8b8ce19ef142" alt=""
data:image/s3,"s3://crabby-images/8d063/8d0637a4cdd8754b7805f4f1b015aeab3828e6fa" alt=""
data:image/s3,"s3://crabby-images/4ccfe/4ccfecabf0817f27f91dc1a7a0a68bb65b06d7ee" alt=""
虽然是报500,原因是我断点导致的超时,不打断点正常调用,是没问题的。至于为什么会调用到这里来,后面详细讲,因为这才是我要讲的重点。
④ DubboBootstrap#checkGlobalConfigs()方法,校验的逻辑,非核心流程,跳过。
⑤ DubboBootstrap#startMetadataCenter()方法:
data:image/s3,"s3://crabby-images/12ff8/12ff86a3e78ed1b14587f8d4a9dbac8a1eff9b3c" alt=""
data:image/s3,"s3://crabby-images/0a21d/0a21d53dcb13eb52a1545d03a79f6d510bfff840" alt=""
说白了就是获取元数据的配置信息,通过调用MetadataReportFactory#getMetadataReport()方法,获取MetadataReport对象,并放入MetadataReportInstance的静态属性metadataReports,缓存起来,供后续使用。
元数据中心是dubbo2.7版本之后新增的功能,主要是为了减轻注册中心的压力,将部分存储在注册中心的内容放到元数据中心。元数据中心的数据只是给自己使用的,改动不需要告知对端,比如服务端修改了元数据,不需要通知消费端。这样注册中心存储的数据减少,同时大大降低了因为配置修改导致注册中心频繁通知监听者,从而大大减轻注册中心的压力。因此也可以在元数据中配置注册中心中心。当然要看到其效果,要先注释掉注册中心相关的配置,代码如下:
data:image/s3,"s3://crabby-images/d09f2/d09f2b9a84f8417f1a3cd62c25230556d4d93390" alt=""
除此之外,还要加上依赖,如下:
XML
<dependency>
<groupId>org.apache.dubbo</groupId>
<artifactId>dubbo-metadata-report-nacos</artifactId>
<version>2.7.10</version>
</dependency>
然后启动服务提供方服务,看看结果:
data:image/s3,"s3://crabby-images/5ddf7/5ddf75e72d9a9519b29e967fc034ae49f1306881" alt=""
data:image/s3,"s3://crabby-images/fbf77/fbf77e2acd992a11b2290ec94c0be5ea1fdff701" alt=""
⑥ DubboBootstrap#startMetadataCenter()方法:
data:image/s3,"s3://crabby-images/5d3c3/5d3c3d002832eebd3bbd56fc40fb65d9206cfba5" alt=""
⑦ DubboBootstrap#initEventListener()方法:
data:image/s3,"s3://crabby-images/0197a/0197ade302b94f9a05559af334ece75b42c660e7" alt=""
data:image/s3,"s3://crabby-images/9ff39/9ff39db36a5c0c36bfd9786a25f68c1359150ac8" alt=""
到这里为止,DubboBootstrap#initialize()方法算是讲完了,回到DubboBootstrap#start()方法,代码如下:
data:image/s3,"s3://crabby-images/05aa6/05aa647b1994979d512966b235e1827cba0b0086" alt=""
再看看DubboBootstrap#exportServices()方法,代码如下:
data:image/s3,"s3://crabby-images/ff717/ff7175e803e23c399110a68a93ed2f20bfa5e20d" alt=""
从ConfigManager中取出所有ServiceConfigBase对象,也就是ServiceBean,看看类的继承层级:
data:image/s3,"s3://crabby-images/37e9b/37e9b502e451ca14628a71a8e8d7a5ef9fe94c47" alt=""
ServiceBean是什么时候被放入ConfigManager中的呢?也就是我前面说的,在这里被放入的,代码如下:
data:image/s3,"s3://crabby-images/184b8/184b84e7a8d68d41bedfb4b0b2fbd97182c68806" alt=""
回到DubboBootstrap#exportServices()方法,遍历获取到的所有ServiceConfig,调用ServiceConfig#export()方法,代码如下:
data:image/s3,"s3://crabby-images/cdfa4/cdfa44d3883961f3686cd01c7a53f17d74a6cef8" alt=""
看看ServiceConfig#checkAndUpdateSubConfigs()方法,代码如下(代码较长,只截取核心):
data:image/s3,"s3://crabby-images/72533/72533e5ab89be33718b658ae68ab6b15572ab552" alt=""
data:image/s3,"s3://crabby-images/bbd3e/bbd3e50665830602fede2c98acdcd6660942a47f" alt=""
data:image/s3,"s3://crabby-images/abffd/abffd83d131f51eeea5b46a612e26366b644be5e" alt=""
data:image/s3,"s3://crabby-images/62e09/62e09b9286e6a57bd8fe9270b5c09d73bfcf9a8c" alt=""
data:image/s3,"s3://crabby-images/e9150/e9150eb0e217c6273f6b286d372a77f44c41e660" alt=""
到这里为止,已经给ServiceConfig对象设置好了协议的配置和注册中心的配置,这部分配置也是我在DubboConfiguration中配置好了。当然也包括接口,也设置好了,主要是通过interfaceName属性,基于类加载器加载并设置的,而interfaceName属性也是之前在ServiceClassPostProcessor中设置的,具体代码为:
data:image/s3,"s3://crabby-images/a00e4/a00e42263fa63a429c1b9b4c702277610c2f6d3d" alt=""
data:image/s3,"s3://crabby-images/ef080/ef0804024fe27916c989a62aac97c8ef40a506b0" alt=""
再看看ServiceBean的 ref属性是如何赋值的,这就需要回到ServiceClassPostProcessor#buildServiceBeanDefinition()方法,代码如下:
data:image/s3,"s3://crabby-images/1f6e6/1f6e6cd48a230d86a5e07286e4cb03524a272af5" alt=""
data:image/s3,"s3://crabby-images/5741c/5741c78ac341af18ca1780d7e1ac9ce4489c719f" alt=""
data:image/s3,"s3://crabby-images/26ad7/26ad76437622cda7da9130d705ad28c2b7477c68" alt=""
重点就是给ref设置的属性为RuntimeBeanReference对象,并且名字为"helloServiceImpl",RuntimeBeanReference是Spring提供的,最终给ref属性注入的对象那个肯定不是RuntimeBeanReference,而是在这里做了处理,代码如下(Spring属性注入的部分源码):
data:image/s3,"s3://crabby-images/d1f83/d1f83cdbf530578f9d4eabc61abee21fa6824f7e" alt=""
data:image/s3,"s3://crabby-images/36453/36453f044e1986614fea7dfaaaa59f323a68e73e" alt=""
data:image/s3,"s3://crabby-images/a3ca7/a3ca71284df1db3ba484fc9e84fbdde51759ae72" alt=""
有这个调用链路可知,最终获取的是RuntimeBeanReference的beanName,通过beanName从Spring容器中拿到HelloServiceImpl对象,通过反射,赋值给ServiceBean的ref属性。打断点看看,结果如下:
data:image/s3,"s3://crabby-images/c4e47/c4e47b9c882bd8cc390523b9051907ea47e1ba69" alt=""
重点来了,也就是ServiceConfig#doExport()方法,代码如下:
data:image/s3,"s3://crabby-images/f600d/f600de9d89c8ff5aff86b23286bc0bc002a4cbbc" alt=""
data:image/s3,"s3://crabby-images/70f55/70f557cc87b2a9bd8ad765dfbdcf8c0f8c30e50c" alt=""
data:image/s3,"s3://crabby-images/2fff5/2fff5905d66e8d33920959de46264da605589b6c" alt=""
回到ServiceConfig#doExportUrls()方法,遍历所有的协议,代码如下:
data:image/s3,"s3://crabby-images/b0f91/b0f91082b0fb079d7636e0241f3c4ec40b91c8f9" alt=""
如果我配置了,是这样的效果:
data:image/s3,"s3://crabby-images/c65aa/c65aaf7f43edddc79bd73bf226ffafb5610097c2" alt=""
data:image/s3,"s3://crabby-images/49116/49116672525ee37f2fc450866e7a3bbf258cde0a" alt=""
data:image/s3,"s3://crabby-images/b33aa/b33aa948dbd7ce6b2dc4cabaa85129707e9ee9a2" alt=""
针对group、version,是可以起到隔离服务的作用的,比如version,如果服务提供方有多个结点的话,可以在服务消费方指定version,服务消费反感发起请求,就会@DubboReference上配置的version找对应的节点,如果找到的节点有多个的话,则会通过负载均衡找到一个节点。再看看ServiceRepository#registerService()方法,代码如下:
data:image/s3,"s3://crabby-images/02bc9/02bc98b5da3bd4d3795353b6c4faa6d80a7f79c4" alt=""
再看看ServiceConfig#doExportUrlsFor1Protocol()方法(核心),代码如下(以打断点为主,涉及到的参数太多了,不可能一个个去看):
java
private void doExportUrlsFor1Protocol(ProtocolConfig protocolConfig, List<URL> registryURLs) {
String name = protocolConfig.getName();
if (StringUtils.isEmpty(name)) {
name = DUBBO;
}
// 将配置的参数设置到map中,没有配置的,设置默认值
Map<String, String> map = new HashMap<String, String>();
map.put(SIDE_KEY, PROVIDER_SIDE);
ServiceConfig.appendRuntimeParameters(map);
AbstractConfig.appendParameters(map, getMetrics());
AbstractConfig.appendParameters(map, getApplication());
AbstractConfig.appendParameters(map, getModule());
// remove 'default.' prefix for configs from ProviderConfig
// appendParameters(map, provider, Constants.DEFAULT_KEY);
AbstractConfig.appendParameters(map, provider);
AbstractConfig.appendParameters(map, protocolConfig);
AbstractConfig.appendParameters(map, this);
MetadataReportConfig metadataReportConfig = getMetadataReportConfig();
if (metadataReportConfig != null && metadataReportConfig.isValid()) {
map.putIfAbsent(METADATA_KEY, REMOTE_METADATA_STORAGE_TYPE);
}
if (CollectionUtils.isNotEmpty(getMethods())) {
for (MethodConfig method : getMethods()) {
AbstractConfig.appendParameters(map, method, method.getName());
String retryKey = method.getName() + ".retry";
if (map.containsKey(retryKey)) {
String retryValue = map.remove(retryKey);
if ("false".equals(retryValue)) {
map.put(method.getName() + ".retries", "0");
}
}
List<ArgumentConfig> arguments = method.getArguments();
if (CollectionUtils.isNotEmpty(arguments)) {
for (ArgumentConfig argument : arguments) {
// convert argument type
if (argument.getType() != null && argument.getType().length() > 0) {
Method[] methods = interfaceClass.getMethods();
// visit all methods
if (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())) {
Class<?>[] argtypes = methods[i].getParameterTypes();
// one callback in the method
if (argument.getIndex() != -1) {
if (argtypes[argument.getIndex()].getName().equals(argument.getType())) {
AbstractConfig.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
for (int j = 0; j < argtypes.length; j++) {
Class<?> argclazz = argtypes[j];
if (argclazz.getName().equals(argument.getType())) {
AbstractConfig.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) {
AbstractConfig.appendParameters(map, argument, method.getName() + "." + argument.getIndex());
} else {
throw new IllegalArgumentException("Argument config must set index or type attribute.eg: <dubbo:argument index='0' .../> or <dubbo:argument type=xxx .../>");
}
}
}
} // end of methods for
}
if (ProtocolUtils.isGeneric(generic)) {
map.put(GENERIC_KEY, generic);
map.put(METHODS_KEY, ANY_VALUE);
} else {
String revision = Version.getVersion(interfaceClass, version);
if (revision != null && revision.length() > 0) {
map.put(REVISION_KEY, revision);
}
String[] methods = Wrapper.getWrapper(interfaceClass).getMethodNames();
if (methods.length == 0) {
logger.warn("No method found in service interface " + interfaceClass.getName());
map.put(METHODS_KEY, ANY_VALUE);
} else {
map.put(METHODS_KEY, StringUtils.join(new HashSet<String>(Arrays.asList(methods)), ","));
}
}
/**
* Here the token value configured by the provider is used to assign the value to ServiceConfig#token
*/
if (ConfigUtils.isEmpty(token) && provider != null) {
token = provider.getToken();
}
if (!ConfigUtils.isEmpty(token)) {
if (ConfigUtils.isDefault(token)) {
map.put(TOKEN_KEY, UUID.randomUUID().toString());
} else {
map.put(TOKEN_KEY, token);
}
}
//init serviceMetadata attachments
serviceMetadata.getAttachments().putAll(map);
// export service
String host = findConfigedHosts(protocolConfig, registryURLs, map); // 获取当前服务的ip地址
Integer port = findConfigedPorts(protocolConfig, name, map);
// dubbo://192.168.217.1:20880/com.szl.service.api.HelloService?anyhost=true&application=service-provider&bind.ip=192.168.217.1
// &bind.port=20880&deprecated=false&dubbo=2.0.2&dynamic=true&generic=false&interface=com.szl.service.api.HelloService&methods=sayHello
// &pid=5252&qos.enable=false&release=&server=netty4&side=provider×tamp=1653840932266
URL url = new URL(name, host, port, getContextPath(protocolConfig).map(p -> p + "/" + path).orElse(path), map);
// You can customize Configurator to append extra parameters
if (ExtensionLoader.getExtensionLoader(ConfiguratorFactory.class)
.hasExtension(url.getProtocol())) {
url = ExtensionLoader.getExtensionLoader(ConfiguratorFactory.class)
.getExtension(url.getProtocol()).getConfigurator(url).configure(url);
}
String scope = url.getParameter(SCOPE_KEY);
// don't export when none is configured
if (!SCOPE_NONE.equalsIgnoreCase(scope)) {
// export to local if the config is not remote (export to remote only when config is remote)
if (!SCOPE_REMOTE.equalsIgnoreCase(scope)) {
exportLocal(url);
}
// export to remote if the config is not local (export to local only when config is local)
if (!SCOPE_LOCAL.equalsIgnoreCase(scope)) {
if (CollectionUtils.isNotEmpty(registryURLs)) {
for (URL registryURL : registryURLs) {
//if protocol is only injvm ,not register
if (LOCAL_PROTOCOL.equalsIgnoreCase(url.getProtocol())) {
continue;
}
url = url.addParameterIfAbsent(DYNAMIC_KEY, registryURL.getParameter(DYNAMIC_KEY));
URL monitorUrl = ConfigValidationUtils.loadMonitor(this, registryURL);
if (monitorUrl != null) {
url = url.addParameterAndEncoded(MONITOR_KEY, monitorUrl.toFullString());
}
if (logger.isInfoEnabled()) {
if (url.getParameter(REGISTER_KEY, true)) {
logger.info("Register dubbo service " + interfaceClass.getName() + " url " + url + " to registry " + registryURL);
} else {
logger.info("Export dubbo service " + interfaceClass.getName() + " to url " + url);
}
}
// For providers, this is used to enable custom proxy to generate invoker
String proxy = url.getParameter(PROXY_KEY);
if (StringUtils.isNotEmpty(proxy)) {
registryURL = registryURL.addParameter(PROXY_KEY, proxy);
}
// 生成 invoker对象
Invoker<?> invoker = PROXY_FACTORY.getInvoker(ref, (Class) interfaceClass, registryURL.addParameterAndEncoded(EXPORT_KEY, url.toFullString()));
DelegateProviderMetaDataInvoker wrapperInvoker = new DelegateProviderMetaDataInvoker(invoker, this);
Exporter<?> exporter = PROTOCOL.export(wrapperInvoker);
exporters.add(exporter);
}
} else {
if (logger.isInfoEnabled()) {
logger.info("Export dubbo service " + interfaceClass.getName() + " to url " + url);
}
// 生成代理对象
Invoker<?> invoker = PROXY_FACTORY.getInvoker(ref, (Class) interfaceClass, url);
DelegateProviderMetaDataInvoker wrapperInvoker = new DelegateProviderMetaDataInvoker(invoker, this);
Exporter<?> exporter = PROTOCOL.export(wrapperInvoker);
exporters.add(exporter);
}
MetadataUtils.publishServiceDefinition(url);
}
}
this.urls.add(url);
}
我先给@DubboService注解添加以下信息,如下:
data:image/s3,"s3://crabby-images/c6736/c6736edd3e0bfab1c2e08a7ac89d5179062292b5" alt=""
data:image/s3,"s3://crabby-images/40415/40415c66b74d516ffe429983430930eefaa1c545" alt=""
data:image/s3,"s3://crabby-images/da341/da3419b95e780f2bbee88ca2bfcf9b0e823e5295" alt=""
data:image/s3,"s3://crabby-images/14c39/14c39fba9730cea27d189378d662883113505e6b" alt=""
因此可以知道,如果不设置scope,则会暴露对外两个地址,一个是本地(测试用,无需调用远程服务),使用的是"injvm"协议;一个是远程,默认使用的是"dubbo"协议。继续往下看:
data:image/s3,"s3://crabby-images/9b28f/9b28f0b2660c76369799d93fd246a87bb85c27cf" alt=""
上图中漏了一点:在调用PROXY_FACTORY.getInvoker()方法中,又执行了:registryURL.addParameterAndEncoded(EXPORT_KEY, url.toFullString())方法,该方法就是将url进行编码,并将其加入到registryURL的export属性中,如下:
data:image/s3,"s3://crabby-images/d2603/d260369ee0a1c9fd4bcd9bb61f173e143e65fd05" alt=""
限于篇幅,先讲到这里,剩下的内容将在下一篇博客中继续讲解,如有错误,恳请指正,谢谢!