环境:maven 工程 jdk 1.8.0_45 maven :3.0.5 tomcat:8.0.23
问题描述:
在windows开发环境中打的war包没问题,在linux中打出的包就报错,同样的jdk,同样的maven 百试不爽,不解
现象:
网上搜索是 jar包冲突:
大体意思就是 工程中依赖了(或者是间接依赖)导致引入了2个不同版本的servlet-api的jar包,tomcat8容器里面也有个servlet-api的jar包。
例如 servlet-api:2.4.jar 和 servlet-api:3.01.jar包冲突,getDispatcherType()这个方法在 2.4这个版本没有,而在高版本servlet-api中有,所以报错。
为什么在windows中打的包没有问题呢?
猜测:jar包加载的顺序。
在windows中打的war包 在tomcat中解压加载的时候 恰巧优先加载了 高版本的 servlet-api的jar包
而在linux中打的war包在tomcat中解压加载的时候优先加载了 低版本的 servlet-api的jar包
所以一个报错一个不报错。。。也只能这样解释了。
解决办法:
想办法优先加载正确的jar包,没有办法就干掉低版本的jar包
直接依赖的可以按如下方式避免在发布的时候把servlet-api包拷到lib目录下
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.0.1</version>
<scope>provided</scope>
</dependency>
间接依赖的可以按如下踢出冲突
<dependency>
<groupId>jcifs</groupId>
<artifactId>jcifs</artifactId>
<version>1.3.17</version>
<exclusions>
<exclusion>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
</exclusion>
</exclusions>
</dependency>
知识点:
可以用scope标签,值设为provided。如下:
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.1</version>
<scope>provided</scope>
<classifier />
</dependency>
scope的其他参数如下:
compile
默认的scope,表示 dependency 都可以在生命周期中使用。而且,这些dependencies 会传递到依赖的项目中。适用于所有阶段,会随着项目一起发布
provided
跟compile相似,但是表明了dependency 由JDK或者容器提供,例如Servlet AP和一些Java EE APIs。这个scope 只能作用在编译和测试时,同时没有传递性。????????
runtime
表示dependency不作用在编译时,但会作用在运行和测试时,如JDBC驱动,适用运行和测试阶段。
test
表示dependency作用在测试时,不作用在运行时。 只在测试时使用,用于编译和运行测试代码。不会随项目发布。
system
跟provided 相似,但是在系统中要以外部JAR包的形式提供,maven不会在repository查找它。