Jar包冲突如何快速解决

在java项目工程中,一般会依赖大量的jar包,经常会出现重复依赖的问题,最常见的则是一个jar包同时依赖了多个版本。自从使用了类似与maven这样的项目构建工具之后,多个版本依赖的问题,得到了很好的解决,maven会帮助我们自动去除多余的版本。但还是有一种情况,会出现非常隐蔽的版本冲突的问题,就是一个第三方的jar包里面已经把它自身需要的依赖jar文件,打包成了同一个jar包,然后我们同时依赖了这个jar,又单独依赖了这个第三方jar所依赖的不同版本的jar文件时,会出现java.lang.NoSuchFieldError,或ClassNotFoundException异常。这样的问题很难从包名分辨出是哪两个jar冲突,同时也绕过了maven的版本控制。下面来详细的描述问题场景:

项目中同时依赖两个jar

<dependency>
   <groupId>com.fasterxml.jackson.core</groupId>
   <artifactId>jackson-annotations</artifactId>
    <version>2.7.4</version>
</dependency>
<dependency>
   <groupId>org.apache.calcite</groupId>
   <artifactId>calcite-avatica</artifactId>
    <version>1.6.0</version>
</dependency>

calcite-avatica-1.6.0.jar里依赖包含了jackson-annotations-2.1.1.jar


jackson-annotations-2.7.4.jar里面的JsonInclude类


这个时候运行代码,会出现以下错误:

Causedby: java.lang.NoSuchFieldError: USE_DEFAULTS

         atcom.fasterxml.jackson.annotation.JsonInclude$Value.<clinit>(JsonInclude.java:205)

         atcom.fasterxml.jackson.databind.cfg.MapperConfig.<clinit>(MapperConfig.java:45)

         atcom.fasterxml.jackson.databind.ObjectMapper.<init>(ObjectMapper.java:543)

         atcom.fasterxml.jackson.databind.ObjectMapper.<init>(ObjectMapper.java:460)

         atorg.springframework.http.converter.json.Jackson2ObjectMapperBuilder.build(Jackson2ObjectMapperBuilder.java:553)

同一个classpath下存在两个版本的jackson-annotations,这个两个版本存在类和字段的删除和新增,而不同的类加载器加载的顺序是不可确定的,这样在实际的运行中,难免会遇到加载错误的类版本。

 

既然出现了版本的冲突,现在介绍下如何快速的找到冲突的jar包文件,移除冲突的版本。

写个打印class的源jar文件的路径的辅助类

/**
 * Created by shaowei on 2017/7/4.
 * 打印出指定class做在的jar包路径
 */
public class PrintClassSource {
    public PrintClassSource() {
        ProtectionDomain pd =JsonInclude.Include.class.getProtectionDomain();
       System.out.println("-------------------:" +pd.getCodeSource().getLocation());
    }
}

如果像上面的报错例子的是使用spirng的web项目,在实例化类的时候报错的,则可以将打印辅助类放到spring的xml文件中进行实例化加载

<beanid="printClassSource"class="cn.sw.study.web.PrintClassSource"></bean>

如果是普通的java应用,则可以将上面代码加到你的执行代码中,打印出来。

-------------------:file:/E:/work/datai/code_study/study-master/study-web/target/study-web/WEB-INF/lib/calcite-avatica-1.6.0.jar

通过上面打印信息则可以看出,此时加载的是calcite-avatica-1.6.0.jar中的JsonInclude.Include类,而USE_DEFAULTS字段是从jackson-annotations的2.6版本开始有这个字段的,calcite-avatica-1.6.0.jar里的jackson-annotations是2.1.1版本,是没有这个字段的。


  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Netty是一个基于Java的网络编程框架,用于快速开发高性能的网络应用程序。在使用Netty时,可能会遇到版本冲突的问题,主要有以下几种情况: 1. 依赖冲突:当项目中引入的多个依赖库中都包含了不同版本的Netty时,就会出现依赖冲突。这可能导致编译错误、运行时异常或功能失效等问题。 2. 兼容性问题:不同版本的Netty可能存在API的变化或行为上的差异,如果在项目中同时使用了不兼容的版本,可能会导致代码无法编译或运行时出现异常。 3. 功能缺失:某些功能可能只在特定版本的Netty中提供,如果使用了不支持该功能的版本,就无法使用相关功能。 为了解决Netty版本冲突的问题,可以采取以下几种方法: 1. 升级或降级:如果项目中已经使用了某个版本的Netty,但又需要使用另一个版本的特定功能,可以尝试升级或降级Netty版本,以满足项目需求。 2. 排除依赖:如果项目中引入了多个依赖库,其中某些库与Netty存在冲突,可以通过在Maven或Gradle配置文件中排除特定依赖,以解决冲突问题。 3. 统一版本:如果项目中使用了多个依赖库,且它们都依赖于不同版本的Netty,可以尝试找到一个兼容所有依赖库的Netty版本,并将所有依赖库都升级到该版本。 4. 使用Shade插件:如果项目中使用了Apache Maven构建工具,可以使用Maven Shade插件将所有依赖库和它们的依赖项打包到一个可执行的JAR文件中,以避免版本冲突。 5. 源码修改:如果以上方法都无法解决问题,可以考虑修改Netty源码,以适配项目需求或解决特定的版本冲突问题。但这种方法需要谨慎操作,并且需要对Netty框架有一定的了解。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值