关于升级 Dubbo 版本到 2.6.5 后启动失败的“坑”

问题现象

Dubbo从低版本升级到2.6.5版本后,启动失败,报错如下:

05-Mar-2019 16:02:25.204 ?? [RMI TCP Connection(2)-127.0.0.1] org.apache.catalina.core.StandardContext.listenerStart Exception sending context initialized event to listener instance of class org.springframework.web.context.ContextLoaderListener
 java.lang.IllegalStateException: Cannot initialize context because there is already a root application context present - check whether you have multiple ContextLoader* definitions in your web.xml!
    at org.springframework.web.context.ContextLoader.initWebApplicationContext(ContextLoader.java:296)
    at org.springframework.web.context.ContextLoaderListener.contextInitialized(ContextLoaderListener.java:107)
    at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:4727)
    at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5189)
    at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)

解决方案

<b><font color='red'>上终极方案:使用2.6.2以下版本或者2.7.0以上版本的dubbo;</font></b>

具体解决方式需要根据项目的情况解决,提供一些其他方案:

  • 方案1和方案2:适合拥有web.xml的纯xml工程;
  • 方案3和方案4:适合没有web.xmlSpring Boot工程;

拥有Web.xml的项目

方案1:删除自己配置的 ContextLoaderListener

删除 web.xml 中如下的配置:

<listener>
     <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

注意:如果有自定义的Listener继承自ContextLoaderListener也需要删除;

这么做的目的是不需要自己去配置初始化Spring框架,Dubbo2.6.3之后可以“自动”初始化Spring框架;

方案2:关闭 Servlet 3.0 的可插性功能

  1. 将自己的web.xmlxsd升级到3.0;
  2. 配置metadata-complete;
  3. DubboContextInitializer添加到描述文件中;
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1" metadata-complete="true">
         
 <context-param>
        <param-name>contextInitializerClasses</param-name>
        <param-value>org.apache.dubbo.config.spring.initializer.DubboApplicationContextInitializer</param-value>
 </context-param>

</web-app>        

没有Web.xml的Spring Boot工程

Spring Boot工程没有特别好的解决方案,提供两个解决思路:

方案3:添加 web.xml 文件并按照传统配置web.xml

  1. Spring Boot工程改造下,创建webapp/WEB-INF目录并创建web.xml文件;
  2. 按照方案2改造工程;
  3. 主要关闭特性后,很多Spring Boot自动做的需要我们手动在web.xml中配置;

NOTE:如果使用此方案来改造,需要注意自己的Spring Boot项目是否还有其他依赖Servelt 3.0特性的地方,并手动配置到web.xml中;

方案4:阻止dubboListener运行

这个方案也没有绕过添加web.xml的命运,做法如下:

  1. 创建webapp/WEB-INF目录并创建web.xml
  2. web.xml中指定absolute-ordering,仅允许Spring Web的配置生效;
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">

    <display-name>risk-etl</display-name>

    <absolute-ordering>
        <name>spring_web</name>
    </absolute-ordering>

</web-app>

原因和原理分析

观察报错日志,报错位置很明显是Spring框架初始化时的报错,重点是:there is already a root application

这个错误抛出位置位于:Spring-web包的ContextLoader类的initWebApplicationContext 方法。

public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {
        if (servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE) != null) {
            throw new IllegalStateException(
                    "Cannot initialize context because there is already a root application context present - " +
                    "check whether you have multiple ContextLoader* definitions in your web.xml!");
        }
        …………
}

原因很明显,ContextLoader被调用了至少两遍,第二遍报错导致项目初始化失败,其主要的“罪魁祸首”是dubbo包下面的web-fragment.xml

Servlet 3.0的可插特性

Servlet 3.0是随着Java EE 6规范发布的,主要新增特性:

  • 支持异步:Servlet可以支持异步;
  • 新增注解:可以使用注解来配置Servlet/Filter/Listener
  • 可查特性:允许使用web-fragment.xml将一个web.xml拆分到多个包中配置;

支持Servlet 3.0规范的容器,在启动后会扫描工程的jar包,找到符合规范的添加了相关注解的类web-fragment.xml然后跟web.xml的配置合并作为整个项目的初始化配置。

发生原因及解决原理

上述问题的发生原因很明显了:

  1. dubbo2.6.3版本为了实现优雅关机(实际上并不好用)引入了web-fragment.xml注册自己的ContextInitializer;
  2. DubboApplicationContextInitializer通过Spring消息广播机制Context加载完成后调用addShutdownHook()JVM注册一个钩子函数,以便JVM关闭时可以释放一些资源防止内存泄露;
  3. dubbo为了保证自己的ContextInitializer被用到(利用的Spring的机制)在自己的web-fragment.xml顺手配置了一个listener
  4. 容器启动时,如果我们自己在web.xml中配置了ContextLoaderListener(或其子类),我们的Listener一般会被优先调用,完成第一次的Spring Context初始化;
  5. 如果是Spring Boot项目,容器会先调用SpringServletContainerInitializer类的onStartup方法,这个方法内部会初始化Spring Context
  6. 我们的或者Spring BootListener调用完成后会调用dubbolistener,这个时候ContextLoader类会检测到已经初始化了一个Context从而报错,引发项目启动失败;
<web-fragment version="3.0" xmlns="http://java.sun.com/xml/ns/javaee"
              xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
              xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-fragment_3_0.xsd">

    <name>dubbo-fragment</name>

    <ordering>
        <before>
            <others/>
        </before>
    </ordering>

    <context-param>
        <param-name>contextInitializerClasses</param-name>
        <param-value>org.apache.dubbo.config.spring.initializer.DubboApplicationContextInitializer</param-value>
    </context-param>

    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

</web-fragment>

Dubbo在 github 上说明了他为什么要这么做:https://github.com/apache/incubator-dubbo/pull/2126

metadata-complete

这个是Servlet 3.0提供的一个属性,等同一个开关,设置为true则表示web.xml已经提供了全部的配置信息,不需要容器再去各个jar包找配置了,换句话就是:关闭可插特性

absolute-ordering

这个属性是SpringServletContainerInitializer注释里面提供的解决思路。这个属性可以理解为指定web-fragment.xml的加载顺序,和ordering标签的区别是,absolute-ordering仅仅针对我们指定的web-fragment.xml做排序。

总结

轻易升级一个基础框架不是一个好的做法,<b>升级基础框架还是应该关注下当前版本和目标升级版本,框架作者做了些什么事情,出现过什么BUG。</b>

当前的Spring Boot的解决方案并不让人满意,毕竟Spring Boot的无Xml的感觉还是很爽的,为了这个升级引入了web.xml会有一点点不爽。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容