解析spring启动加载dubbo过程
一:简单配置
web.xml
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd">
<!-- 扫描注解 -->
<context:component-scan base-package="per.qiao" />
<!-- 目的:找到注册中心,注册当前服务的基本信息 -->
<!-- 1.配置别名, 目的是在后台可以看到这个服务的别名,好区分到底是谁 -->
<dubbo:application name="serviceImpl"/>
<!-- 2. 配置注册中心 address : 注册中心地址 protocol: 注册中心的协议格式 -->
<dubbo:registry address="192.168.199.247:2181" protocol="zookeeper" />
<!-- 3. 选择需要暴露的方法 interface: 目标类的类型 ref: 目标类的具体实现 timeout: 超时连接时间-->
<dubbo:service interface="per.qiao.service.TestService" ref="serviceImpl" timeout="10000" />
<!-- 配置当前服务暴露的端口 以及暴露协议 -->
<dubbo:protocol name="dubbo" port="9000"/>
</beans>
dubbo的默认文件
spring.handlers文件
http\://dubbo.apache.org/schema/dubbo=com.alibaba.dubbo.config.spring.schema.DubboNamespaceHandler
http\://code.alibabatech.com/schema/dubbo=com.alibaba.dubbo.config.spring.schema.DubboNamespaceHandler
spring.schemas文件
http\://dubbo.apache.org/schema/dubbo/dubbo.xsd=META-INF/dubbo.xsd
http\://code.alibabatech.com/schema/dubbo/dubbo.xsd=META-INF/compat/dubbo.xsd
说明: spring.handlers文件用来配置解析dubbo标签并封装成对应的对象
**spring.schemas文件用来配置schame文件的位置 **
当spring容器扫描到配置文件,比如applicationContext时,遇到名称空间xmlns:dubbo="http://code.alibabatech.com/schema/dubbo",就会通过名称空间去查询对应的xsd约束文件,就如schemaLocation中配置的
- xsi:schemaLocation=http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd
- 再用这个找到的xsd去spring.schemas文件中找到xsd文件的位置,并校验xsd文件的正确性,返回当前文件(applicationContext.xml)的Document对象
二、启动过程
由ContextLoaderListener进入,调用contextInitialized进入initWebApplicationContext开始spring容器
进入configureAndRefreshWebApplicationContext方法, 然后调用refresh方法
refresh中有个obtainFreshBeanFactory方法,进去
走流水线AbstractApplicationContext->AbstractRefreshableApplicationContext#refreshBeanFactory -> XmlWebApplicationContext#loadBeanDefinitions
进入到XmlBeanDefinitionReader#loadBeanDefinitions(configLocation)
// 遍历你contextConfigLocation配置的多个xml文件
for (String configLocation : configLocations) {
reader.loadBeanDefinitions(configLocation);
}
来到AbstractBeanDefinitionReader#loadBeanDefinitions-> XmlBeanDefinitionReader#loadBeanDefinitions, doLoadBeanDefinitions, registerBeanDefinitions
看一下doLoadBeanDefinitions 这里会加载你的spring.schame文件
//这个方法会检查你的文件(eg. applicationContext.xml)中的schame的namespace对应的xsd是否正确,返回当前文件的Document对象
Document doc = doLoadDocument(inputSource, resource);
return registerBeanDefinitions(doc, resource);
这个方法是 注册给定DOM文档中包含的bean定义, 也就是解析你的applicationContext.xml中定义的bean
registerBeanDefinitions
public int registerBeanDefinitions(Document doc, Resource resource) throws... {
//创建新的doc解析器
BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
// 已有的bean个数
int countBefore = getRegistry().getBeanDefinitionCount();
// 读取document中的bean
documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
return getRegistry().getBeanDefinitionCount() - countBefore;
}
- DefaultBeanDefinitionDocumentReader#registerBeanDefinitions
@Override
public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
this.readerContext = readerContext;
Element root = doc.getDocumentElement();
//读取bean
doRegisterBeanDefinitions(root);
}
- BeanDefinitionParserDelegate#parseCustomElement 这里就从容器中调用到自定义的handler中
public BeanDefinition parseCustomElement(Element ele, BeanDefinition containingBd) {
//获取节点的NamespaceURI, 如果当前节点是dubbo:application 那么就是根据dubbo找到
//xmlns:dubbo="http://code.alibabatech.com/schema/dubbo" 后面的uri
String namespaceUri = getNamespaceURI(ele);
//这里的resolve方法,会加载spring.handlers文件,调用namespaceHandler.init();获取当前节点对应的handler解析器
NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
//parse方法会先根据element的名称获取对应的BeanDefinitionParser
//比如当前元素是dubbo:application, 这里会用application作为key去获取对应的解析器,然后调用其parse方法
return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}
- DubboBeanDefinitionParser#parse
关于第8步,加载了spring.handlers文件,然后调用DubboNamespaceHandler的init方法,然后是registerBeanDefinitionParser方法,该方法将节点名称和解析封装在NamespaceHandlerSupport的map中
protected final void registerBeanDefinitionParser(String elementName, BeanDefinitionParser parser) {
this.parsers.put(elementName, parser);
}
然后在handler.parse(...) -> NamespaceHandlerSupport#findParserForElement方法里, 能通过下面的方式获取到
BeanDefinitionParser parser = this.parsers.get(localName);
三、说一个遇到的坑-关于idea的this与toString
在查看解析spring.schames文件的源码时,走到PluggableSchemaResolver类里面,当断点进入resolveEntity方法时,属性private volatile Map<String, String> schemaMappings; 已经有值了,这是不正常的,为此我花了很长时间去找原因
最后发现,是由于idea在debug时,在debugger界面有个this引用着当前对象,它会调用当前类的toString()方法,而且是构造方法执行后每执行一步,它刷新一次(调用toString()),由于toString方法中有调用getSchemaMappings方法,会给你加载数据。 最坑的是,idea调用的用断点拦截不到
你可以尝试着还原这个坑
public class MyTest {
public static void main(String[] args) {
BB b = new BB();
}
static class BB {
private int n = 0;
public BB() {
System.out.println("i am qiao" + n);
}
@Override
public String toString() {
n = 2;
System.out.println("hello");
return "n == " + n;
}
}
}
四、小结:
web.xml中 参数contextConfigLocation的值可以使用,; \t\n(逗号|分号|空格|制表符|换行)分隔开(ContextLoader.INIT_PARAM_DELIMITERS)
classpath:与classpath*: 的区别
classpath: 会从classes目录下获取文件
classpath*: 会从所有路径下加载文件,包括jar包
详见:PathMatchingResourcePatternResolver#getResources
spring.handlers文件用来配置解析dubbo标签的handler
spring.schemas文件用来配置schame文件的位置
PluggableSchemaResolver类解析的spring.shames
DefaultNamespaceHandlerResolver类解析的spring.handler
---恢复内容结束---
解析spring启动加载dubbo过程的更多相关文章
- AngularJS进阶(三十九)基于项目实战解析ng启动加载过程
基于项目实战解析ng启动加载过程 前言 在AngularJS项目开发过程中,自己将遇到的问题进行了整理.回过头来总结一下angular的启动过程. 下面以实际项目为例进行简要讲解. 1.载入ng库 2 ...
- spring启动加载过程源码分析
我们知道启动spring容器两常见的两种方式(其实都是加载spring容器的xml配置文件时启动的): 1.在应用程序下加载 ApplicationContext ctx = new ClassPat ...
- springboot集成mybatis源码分析-启动加载mybatis过程(二)
1.springboot项目最核心的就是自动加载配置,该功能则依赖的是一个注解@SpringBootApplication中的@EnableAutoConfiguration 2.EnableAuto ...
- Spring 启动加载资源到内存
前言 在一些业务场景中,当容器初始化完成之后,需要处理一些操作,比如一些数据的加载.初始化缓存.特定任务的注册等等.我找到了三种方式解决下面的问题. 1.使用PostConstruct注解 这种解决方 ...
- spring启动加载类,手动加载bean
方法一: public final class Assembler implements BeanFactoryPostProcessor { private static ConfigurableL ...
- Tomcat源码分析三:Tomcat启动加载过程(一)的源码解析
Tomcat启动加载过程(一)的源码解析 今天,我将分享用源码的方式讲解Tomcat启动的加载过程,关于Tomcat的架构请参阅<Tomcat源码分析二:先看看Tomcat的整体架构>一文 ...
- Spring源码解析-配置文件的加载
spring是一个很有名的java开源框架,作为一名javaer还是有必要了解spring的设计原理和机制,beans.core.context作为spring的三个核心组件.而三个组件中最重要的就是 ...
- redis启动加载过程、数据持久化
背景 公司一年的部分业务数据放在redis服务器上,但数据量比较大,单纯的string类型数据一年就将近32G,而且是经过压缩后的. 所以我在想能否通过获取string数据的时间改为保存list数据类 ...
- spring boot启动加载项CommandLineRunner
spring boot启动加载项CommandLineRunner 在使用SpringBoot构建项目时,我们通常有一些预先数据的加载.那么SpringBoot提供了一个简单的方式来实现–Comman ...
随机推荐
- Qt的插件开发
写代码都是从不会到会,那么写博客也是同样的道理.从不会到会最实用的办法就是模仿了.关于Qt的知识很多都是学习了CSDN的一位大神 一去二三里.关于Qt插件的开发,我们也从他的文章里面抽丝剥茧,把最本质 ...
- 在CentOS7中安装zookeeper
参考:https://www.linuxidc.com/Linux/2016-09/135052.htm 1.zookeeper运行需要jdk环境,先确保有配置jdk,可以参考此处 2.下载解压zoo ...
- Python3中转换字符串编码
在使用subprocess调用Windows命令时,遇到了字符串不显示中文的问题,源码如下:#-*-coding:utf-8-*-__author__ = '$USER' #-*-coding:utf ...
- html 与 xml 的区别与联系
[引言] 前一阵子刚刚学习了html(HyperText Markup Language),最近又接触了xml(Extensible Markup Language),它们之间有什么联系和区别呢?现在 ...
- C++ STL介绍——简介
目录 1.什么是STL 2.STL中六大组件 2.1 容器(Container) 2.2 迭代器(Iterator) 2.3 算法(Algorithm) 2.4 仿函数(Functor) 2.5 适配 ...
- matlab图像灰度调整——imadjust函数的使用
在MATLAB中,通过函数imadjust()进行图像灰度的调整,该函数调用格式如下: J=imadjust( I ) 对图像I进行灰度调整 J=imadjust( I,[low_in;high_i ...
- android studio: 让项目通过阿里云 maven jcenter 下载依赖资源
打开项目根目录下的 build.gradle(Project:项目名称一级的gradle),如下所示添加阿里 maven 库地址: // Top-level build file where you ...
- android studio: 实现类似于XCode中的#pragma mark的效果
代码行数写多了,想找一个指定的方法真困难,关键有时候记不住方法的名字,用Ctrl+O也不好使,突然想到以前做iOS开发时,XCode里有一个#pragma mark的功能,很好用:在代码中定义这样一个 ...
- 002-创建型-04-建造者模式(Builder)、JDK1.7源码中的建造者模式、Spring中的建造者模式
一.概述 建造者模式的定义:将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 工厂类模式提供的是创建单个类的模式,而建造者模式则是将各种产品集中起来进行管理,用来创建复合对象 ...
- LeetCode_28. Implement strStr()
28. Implement strStr() Easy Implement strStr(). Return the index of the first occurrence of needle i ...