Dubbo中@Service工作过程解析
Dubbo @Service
注解的加载和启动过程
Spring中的BeanPostProcessor
首先我们应当了解到在spring体系中BeanPostProcessor
是什么、加载流程
它是什么
BeanPostProcessor
也也称为后置处理器。在spring容加载流程。
spring容器bean加载流程
1 | // Prepare this context for refreshing. |
后置处理器的加载和工作
所以由上可以看出,在registerBeanPostProcessors
这一步的时候会划分容器中各种后置处理器,首先归类有@Priority
注解,其次归类有@Order
注解。最后划分其它的。1
2
3
4
5
6
7
8
9
10
11
12
13
14// 1
sortPostProcessors(priorityOrderedPostProcessors, beanFactory);
registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors);
// 2
sortPostProcessors(orderedPostProcessors, beanFactory);
registerBeanPostProcessors(beanFactory, orderedPostProcessors);
// 3
registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors);
// Finally, re-register all internal BeanPostProcessors.
sortPostProcessors(internalPostProcessors, beanFactory);
registerBeanPostProcessors(beanFactory, internalPostProcessors);
这样,就将所有的后置处理器注册到容器中。后续在容器启动的过程中,会通过反射的方式调用各个实现了BeanPostProcessor
的实现类的beforexxxx
和afterxxx
的方法来做进一步的处理。
有了以上基础,就可以去dubbo包中找到各自的
xxxBeanPostProcessor
了。
dubbo中的ServiceAnnotationBeanPostProcessor
类继承关系
作用
BeanDefinitionRegistryPostProcessor继承自BeanFactoryPostProcessor,是一种比较特殊的BeanFactoryPostProcessor。BeanDefinitionRegistryPostProcessor中定义的postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry)方法 可以让我们实现自定义的注册bean定义的逻辑。
从以上论述可以看出实现BeanDefinitionRegistryPostProcessor
的作用就是向spring容器中注册响应的bean实例。
具体分析
1 |
|
下面就分析registerServiceBeans
的处理过程。
- 扫描系统配置的
basePackages
,将@Service
注解的类放到一个set集合中。 - 找到所有标注
@Service
的类是否被扫面到。 registerServiceBean
循环遍历这个集合,并将它们注入到spring容器中。
ServiceBean的作用
类图继承关系
主要看它实现了InitializingBean
。通过实现它来对bean初始化之后做一定操作(调用afterPropertiesSet()
)。
代码实现
ServiceBean初始化过程。如下所示:
1
2
3
4
5
6
7"unchecked", "deprecation"}) ({
public void afterPropertiesSet() throws Exception {
// 进行大量的操作,来读取对应的@Service组成相应的对象信息.....
.....
// 最后一步,导出服务。
export()
}export()导出服务
最后得到一系列的URL信息(形如注册在zk上的provider节点信息)
1 | export() -> doExportUrlsFor1Protocol() |
获取暴露的host和端口
1
2
3
4// 获取部署主机信息
String host = this.findConfigedHosts(protocolConfig, registryURLs, map);
// 获取配置的端口信息。(默认20880,)
Integer port = this.findConfigedPorts(protocolConfig, name, map);导出有两步
- exportLocal
- exportJVM
根据第3步启动相关的服务
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
public <T> Exporter<T> export(Invoker<T> invoker) throws RpcException {
//
URL url = invoker.getUrl();
// export service.
String key = serviceKey(url);
DubboExporter<T> exporter = new DubboExporter<T>(invoker, key, exporterMap);
exporterMap.put(key, exporter);
//export an stub service for dispatching event
Boolean isStubSupportEvent = url.getParameter(STUB_EVENT_KEY, DEFAULT_STUB_EVENT);
Boolean isCallbackservice = url.getParameter(IS_CALLBACK_SERVICE, false);
if (isStubSupportEvent && !isCallbackservice) {
String stubServiceMethods = url.getParameter(STUB_EVENT_METHODS_KEY);
if (stubServiceMethods == null || stubServiceMethods.length() == 0) {
if (logger.isWarnEnabled()) {
logger.warn(new IllegalStateException("consumer [" + url.getParameter(INTERFACE_KEY) +
"], has set stubproxy support event ,but no stub methods founded."));
}
} else {
stubServiceMethodsMap.put(url.getServiceKey(), stubServiceMethods);
}
}
// 根据url配置host、port信息启动服务(默认使用netty)
openServer(url);
optimizeSerialization(url);
return exporter;
}netty服务启动
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31private ExchangeServer createServer(URL url) {
url = URLBuilder.from(url)
// send readonly event when server closes, it's enabled by default
.addParameterIfAbsent(CHANNEL_READONLYEVENT_SENT_KEY, Boolean.TRUE.toString())
// enable heartbeat by default
.addParameterIfAbsent(HEARTBEAT_KEY, String.valueOf(DEFAULT_HEARTBEAT))
.addParameter(CODEC_KEY, DubboCodec.NAME)
.build();
String str = url.getParameter(SERVER_KEY, DEFAULT_REMOTING_SERVER);
if (str != null && str.length() > 0 && !ExtensionLoader.getExtensionLoader(Transporter.class).hasExtension(str)) {
throw new RpcException("Unsupported server type: " + str + ", url: " + url);
}
ExchangeServer server;
try {
server = Exchangers.bind(url, requestHandler);
} catch (RemotingException e) {
throw new RpcException("Fail to start server(url: " + url + ") " + e.getMessage(), e);
}
str = url.getParameter(CLIENT_KEY);
if (str != null && str.length() > 0) {
Set<String> supportedTypes = ExtensionLoader.getExtensionLoader(Transporter.class).getSupportedExtensions();
if (!supportedTypes.contains(str)) {
throw new RpcException("Unsupported client type: " + str);
}
}
return server;
}