在 spring boot 出来之前,或者没有使用 spring boot 时,Java EE 开发时如果选择 tomcat servlet,需要自己指定 tomcat 版本;此处没有考虑那种直接把打包的 war 直接扔到本地安装的任意版本的 tomcat,然后启动外置 tomcat 的情况。
使用 spring boot (内置 tomcat)时,一般情况下,完全没有必要去修改 tomcat 的版本,就算是生产环境使用的 spring boot 内置版本,也是最好使用 spring boot 的内置的 tomcat 版本,这是经过兼容测试、回归测试的版本号。
但是,很多公司都不推荐各个应用使用内置 tomcat 这种方式,因为这样的话,100 个服务,可能有几十种版本的 tomcat,不方便统一管理,太老的 tomcat 版本有问题,太新的版本不稳定。所以很多公司仍然在使用经过很多公司多年生产环境校验的 tomcat 7,甚至是 tomcat 6。随着版本的提升,依赖于 tomcat 的良好架构设计,其后续版本的各种性能提升,对于 http 2的支持,难道你真的舍得不去尝试使用吗?
不过,本地开发时,完全可以随意使用最新版本的 tomcat,如果是使用 spring boot,并且是使用内置 tomcat 的话,直接使用最新版本的 sb 即可,因为每次 sb 升级,都会升级 tomcat 版本。但是最新版本 sb 可能有不兼容等问题,有些人可能不会去尝试使用。
如果不是使用 spring boot 内置 tomcat,而是想使用任意版本的 tomcat 呢?
下来详细讲讲怎么修改 tomcat 的版本吧,也是去试着理解一下 tomcat 的启动过程,以及 spring boot 如何内嵌集成 tomcat 的。
想要修改内置的 tomcat 的默认版本,首先得知道当前是什么版本。
比如我当前的 sb 版本是:
<parent>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-parentartifactId>
<version>1.5.7.RELEASEversion>
parent>
借助于 IDE,查看 spring-boot-starter-tomcat 的 pom 文件:
<dependencies>
<dependency>
<groupId>org.apache.tomcat.embedgroupId>
<artifactId>tomcat-embed-coreartifactId>
dependency>
<dependency>
<groupId>org.apache.tomcat.embedgroupId>
<artifactId>tomcat-embed-elartifactId>
dependency>
<dependency>
<groupId>org.apache.tomcat.embedgroupId>
<artifactId>tomcat-embed-websocketartifactId>
dependency>
dependencies>
可知所有的 jar。
此时如果是 eclipse 的忠实用户,则可以在当前工程 project 的 maven 依赖里面直接去看 jar 的版本号。
IDEA 则略有不同,IDEA 的 module 概念等于 eclipse 的 project 概念。IDEA 在 External Libraries下面并不能看到版本号信息,尤其是对于多个 module 的工程 project 而言,不同 module 使用不同的 tomcat 版本时如何知道当前 module 使用的是什么版本的 tomcat 呢?
实际上,此时就是要去理解 IDEA 的设计意图咯,我认为 External Libraries 的作用主要是去看源码,debug。在右侧还有一个面板 Maven Projects,找到当前 module,打开 dependencies 信息,然后此时还可以查看 jar 包,即 artifactId 的依赖管理关系:
可知 spring boot 1.5.7 的内置 tomcat 版本是 8.5.20。
打开文件:
/Users/awesome-me/.m2/repository/org/springframework/boot/spring-boot-dependencies/1.5.7.RELEASE/spring-boot-dependencies-1.5.7.RELEASE.pom
可以在标签
下面找到
从上面的方法2,就知道如何修改的思路。
即在 pom.xml 文件里面添加一个标签
,添加期望的版本
如何知道修改是否成功?
修改成
启动信息:
2018-03-15 00:46:26.275 INFO 47112 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat initialized with port(s): 8080 (http)
2018-03-15 00:46:26.282 INFO 47112 --- [ main] o.apache.catalina.core.StandardService : Starting service Tomcat
2018-03-15 00:46:26.283 INFO 47112 --- [ main] org.apache.catalina.core.StandardEngine : Starting Servlet Engine: Apache Tomcat/8.0.30
2018-03-15 00:46:26.333 INFO 47112 --- [ost-startStop-1] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext
2018-03-15 00:46:26.333 INFO 47112 --- [ost-startStop-1] o.s.web.context.ContextLoader : Root WebApplicationContext: initialization completed in 1000 ms
但是,有时候启动会报错:
Caused by: java.lang.NoClassDefFoundError: org/apache/juli/logging/LogFactory
at org.apache.catalina.util.LifecycleBase.(LifecycleBase.java:37)
at org.springframework.boot.context.embedded.tomcat.TomcatEmbeddedServletContainerFactory.getEmbeddedServletContainer(TomcatEmbeddedServletContainerFactory.java:169)
at org.springframework.boot.context.embedded.EmbeddedWebApplicationContext.createEmbeddedServletContainer(EmbeddedWebApplicationContext.java:164)
at org.springframework.boot.context.embedded.EmbeddedWebApplicationContext.onRefresh(EmbeddedWebApplicationContext.java:134)
... 13 common frames omitted
显然是缺少 jar 包,在 dependency 里面添加即可。
<dependency>
<groupId>org.apache.tomcatgroupId>
<artifactId>tomcat-juliartifactId>
<version>${tomcat.version}version>
dependency>
但是还有另外一个 artifactId
<dependency>
<groupId>org.apache.tomcat.embedgroupId>
<artifactId>tomcat-embed-logging-juliartifactId>
<version>${tomcat.version}version>
dependency>
使用下面这个更好?(待验证)
但是,还有一个问题:
spring-boot-starter-web
是包含spring-boot-starter-tomcat
的,查看 pom 文件可知。
也就是说,我们没有必要重复添加spring-boot-starter-tomcat
,一个spring-boot-starter-web
就可以把一个典型的 spring web 项目搭建成功。也方便了 jar 包的管理。
但是,如果此时你的项目中仅仅添加spring-boot-starter- web
,而没有直接添加spring-boot-starter-tomcat
,就算配置
<properties>
<tomcat.version>9.0.6tomcat.version>
properties>
也并不能生效。至于为什么,有待研究 spring boot 的 autoconfigure 源码才能知道吧。