Spring在启动Dubbo服务端应用时,会实例化ServiceBean<T>并设置配置属性,然后调用export方法:

    @SuppressWarnings({"unchecked", "deprecation"})
public void afterPropertiesSet() throws Exception {
// 设置一揽子provider属性
...... if (!isDelay()) {
export();
}
}

此后调用的是ServiceConfig中的doExportUrls方法:

    @SuppressWarnings({"unchecked", "rawtypes"})
private void doExportUrls() {
// 获取注册中心地址
List<URL> registryURLs = loadRegistries(true);
for (ProtocolConfig protocolConfig : protocols) {
doExportUrlsFor1Protocol(protocolConfig, registryURLs);
}
}

最终实现的是如下逻辑:

    private void doExportUrlsFor1Protocol(ProtocolConfig protocolConfig, List<URL> registryURLs) {
// 省略...
...... String scope = url.getParameter(Constants.SCOPE_KEY);
// don't export when none is configured
if (!Constants.SCOPE_NONE.toString().equalsIgnoreCase(scope)) { // export to local if the config is not remote (export to remote only when config is remote)
if (!Constants.SCOPE_REMOTE.toString().equalsIgnoreCase(scope)) {
// 暴露到本地,使用injvm协议,本地调用服务时会采用此协议
exportLocal(url);
}
// export to remote if the config is not local (export to local only when config is local)
if (!Constants.SCOPE_LOCAL.toString().equalsIgnoreCase(scope)) {
// 暴露到服务注册中心
if (logger.isInfoEnabled()) {
logger.info("Export dubbo service " + interfaceClass.getName() + " to url " + url);
}
if (registryURLs != null && !registryURLs.isEmpty()) {
for (URL registryURL : registryURLs) {
url = url.addParameterIfAbsent("dynamic", registryURL.getParameter("dynamic"));
URL monitorUrl = loadMonitor(registryURL);
if (monitorUrl != null) {
url = url.addParameterAndEncoded(Constants.MONITOR_KEY, monitorUrl.toFullString());
}
if (logger.isInfoEnabled()) {
logger.info("Register dubbo service " + interfaceClass.getName() + " url " + url + " to registry " + registryURL);
}
Invoker<?> invoker = proxyFactory.getInvoker(ref, (Class) interfaceClass, registryURL.addParameterAndEncoded(Constants.EXPORT_KEY, url.toFullString()));
DelegateProviderMetaDataInvoker wrapperInvoker = new DelegateProviderMetaDataInvoker(invoker, this); // 暴露服务,protocol实例为ProtocolListenerWrapper,ProtocolListenerWrapper使用装饰器模式,会不断的调用Protocol接口的子类实例
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);
}
}
}
this.urls.add(url);
}

代码 Exporter<?> exporter = protocol.export(wrapperInvoker); protocol实例为ProtocolListenerWrapper,ProtocolListenerWrapper使用了装饰器模式,依次包装了Protocol接口的子类,其中最重要的两个子类是DubboProtocol和RegistryProtocol。DubboProtocol的主要功能是启动NettyServer和重置服务URL的部分参数,RegistryProtocol的主要功能是暴露服务到注册中心,例如zookeeper。

接下来先看DubboProtocol的export方法:

    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(Constants.STUB_EVENT_KEY, Constants.DEFAULT_STUB_EVENT);
Boolean isCallbackservice = url.getParameter(Constants.IS_CALLBACK_SERVICE, false);
if (isStubSupportEvent && !isCallbackservice) {
String stubServiceMethods = url.getParameter(Constants.STUB_EVENT_METHODS_KEY);
if (stubServiceMethods == null || stubServiceMethods.length() == 0) {
if (logger.isWarnEnabled()) {
logger.warn(new IllegalStateException("consumer [" + url.getParameter(Constants.INTERFACE_KEY) +
"], has set stubproxy support event ,but no stub methods founded."));
}
} else {
stubServiceMethodsMap.put(url.getServiceKey(), stubServiceMethods);
}
} // 启动NettyServer和重置服务URL的部分参数
openServer(url);
optimizeSerialization(url);
return exporter;
}
    private void openServer(URL url) {
// find server.
String key = url.getAddress();
//client can export a service which's only for server to invoke
boolean isServer = url.getParameter(Constants.IS_SERVER_KEY, true);
if (isServer) {
ExchangeServer server = serverMap.get(key);
if (server == null) {
// 启动NettyServer并放入serverMap
serverMap.put(key, createServer(url));
} else {
// server supports reset, use together with override
// 重置服务URL的部分参数
server.reset(url);
}
}
}
    private ExchangeServer createServer(URL url) {

        ......

        url = url.addParameter(Constants.CODEC_KEY, DubboCodec.NAME);
ExchangeServer server;
try {
// 启动NettyServer
server = Exchangers.bind(url, requestHandler);
} catch (RemotingException e) {
throw new RpcException("Fail to start server(url: " + url + ") " + e.getMessage(), e);
} ...... return server;
}

最终会调用HeaderExchanger的bind方法:

public ExchangeServer bind(URL url, ExchangeHandler handler) throws RemotingException {
return new HeaderExchangeServer(Transporters.bind(url, new DecodeHandler(new HeaderExchangeHandler(handler))));
}

剩下就是实例化NettyServer了,这里不再贴出该部分代码,若感兴趣可自行跟踪源码啦。

这里注意到NettyServer中有一个属性channels,这里维护了所有客户端的连接。

最新文章

  1. [转]Android开发:Parallax效果的ScrollerView,改编自ParallaxListView
  2. poj-3739. Special Squares(二维前缀和)
  3. js数组到后台转 list数组
  4. HTML知识点总结以及典型例子讲解
  5. SqlSugar简单工模式数据访问简单Demo
  6. windows和linux双系统删除linux
  7. samba后台进程及安全模式简介
  8. Sublime Text Packages Control 安装
  9. SpringMVC配置+小例子
  10. Hadoop它——跑start-all.sh时间namenode不启动
  11. Flex布局介绍
  12. java读取properties中文乱码
  13. UVA12298 Super Poker II
  14. BZOJ2480Spoj3105 Mod&amp;BZOJ1467Pku3243 clever Y——EXBSGS
  15. linux学习第十二天 (Linux就该这么学)找到一本不错的Linux电子书,附《Linux就该这么学》章节目录
  16. React 入门学习笔记整理(七)—— 生命周期
  17. C#获取驱动器盘符
  18. 黄聪:C# webBrowser控件禁用alert,confirm之类的弹窗解决方案
  19. STM32中TIMx的映射及其通道
  20. 我的Pycharm,我做主

热门文章

  1. 借助 Filter 生成静态页面缓存问题
  2. 文件下载(Servlet/Struts2)
  3. Access MetaData
  4. 自己写的一个nodejs查找文件模块-node-find-all-files
  5. WC前的颓废——带花树
  6. Maven学习总结(七):Maven的聚合和继承
  7. Child extends Parent,可以得到什么?
  8. android 获取http请求json数据
  9. Ubuntu 16.04 Apache2 更改访问html根路径方案(可解决403)
  10. PyCharm添加Selenium与Appium依赖