如何避免Java项目中的循环依赖问题

开发 后端
当我们开发一个Java项目时,我们要做的第一件事就是添加我们需要使用的依赖项,库或框架。为此,我们使用依赖管理工具(例如Maven或Gradle),来管理我们的项目依赖。

[[384504]]

 前言

当我们开发一个Java项目时,我们要做的第一件事就是添加我们需要使用的依赖项,库或框架。为此,我们使用依赖管理工具(例如Maven或Gradle),来管理我们的项目依赖。

我们为项目添加的每个依赖项都包含其他依赖项。我们都知道依赖传递这个概念。在许多情况下,项目的依赖传递不会有任何的问题,因为依赖管理工具(Maven或Gradle)很好地解决了不同库之间的版本冲突。但是,在其他情况下,当使用我们的API /库时或者不同的库依赖的同一个库不同版本,这些冲突会产生一些错误。

举个例子:


图中我们的API 库导入了一些库,但这些库同时依赖了库X的不同版本,这样就好出现依赖冲突。

如何解决这个问题

在没有Maven & Gradle 库管理工具之前,我们开发一个Java项目需要耗费大量的时间去解决我们项目的依赖冲突,Maven & Gradle工具的引入极大的解决了我们依赖的问题,但很多时候我们还是会看到诸如:ClassNotFoundException,MethodNotSupportedException,NoClassDefNotFound这个时候很多情况下依赖开发人员的经验手动去解决,同时我们也可以依赖一些开发工具去帮助我们排查问题,如:

Eclipse / IntelliJ 开发工具中的Dependency Analyzer

他可以展示所有的依赖项并协助开发人员定位到有冲突的库。


当您检测哪些库包含导致问题的依赖项时,需要从我们的依赖项管理文件(pom.xml或build.gradle)中排除所有包含不同版本依赖项的库。

maven-enforcer-plugin插件

Maven提供了Maven-Enforcer-Plugin插件,用来校验约定遵守情况(或者说校验开发环境)。比如JDK的版本,Maven的版本,开发环境(Linux,Windows等),依赖jar包的版本等等

插件使用只需要在pom中引入即可:

  1. <plugin> 
  2.     <groupId>org.apache.maven.plugins</groupId> 
  3.     <artifactId>maven-enforcer-plugin</artifactId> 
  4.     <version>1.3.1</version> 
  5.     <executions> 
  6.         <execution> 
  7.             <id>enforce</id> 
  8.             <configuration> 
  9.                 <rules> 
  10.                     <DependencyConvergence/> 
  11.                 </rules> 
  12.             </configuration> 
  13.             <goals> 
  14.                 <goal>enforce</goal> 
  15.             </goals> 
  16.         </execution> 
  17.     </executions> 
  18. </plugin> 

在进行mvn clean package的时候,会在console中打印出来冲突的jar版本和其父pom,如下:

  1. [INFO] --- maven-enforcer-plugin:1.3.1:enforce (enforce) @ uaf-uafei-provider --- 
  2. [WARNING]  
  3. Dependency convergence error for com.google.guava:guava:16.0 paths to dependency are: 
  4. +-com.uaf.uafei:uaf-uafei-provider:0.2.09-SNAPSHOT 
  5.   +-com.uaf:microservice-sleuth:0.2.09-SNAPSHOT 
  6.     +-org.springframework.cloud:spring-cloud-starter-openfeign:2.1.5.RELEASE 
  7.       +-io.github.openfeign:feign-hystrix:10.4.0 
  8.         +-com.netflix.archaius:archaius-core:0.7.6 
  9.           +-com.google.guava:guava:16.0 
  10. and 
  11. +-com.uaf.uafei:uaf-uafei-provider:0.2.09-SNAPSHOT 
  12.   +-com.uaf:microservice-apollo:0.2.09-SNAPSHOT 
  13.     +-com.google.inject:guice:4.1.0 
  14.       +-com.google.guava:guava:19.0 
  15.  
  16. [WARNING] Rule 0: org.apache.maven.plugins.enforcer.DependencyConvergence failed with message: 
  17. Failed while enforcing releasability the error(s) are [ 
  18. Dependency convergence error for com.google.guava:guava:16.0 paths to dependency are: 
  19. +-com.uaf.uafei:uaf-uafei-provider:0.2.09-SNAPSHOT 
  20.   +-com.uaf:microservice-sleuth:0.2.09-SNAPSHOT 
  21.     +-org.springframework.cloud:spring-cloud-starter-openfeign:2.1.5.RELEASE 
  22.       +-io.github.openfeign:feign-hystrix:10.4.0 
  23.         +-com.netflix.archaius:archaius-core:0.7.6 
  24.           +-com.google.guava:guava:16.0 
  25. and 
  26. +-com.uaf.uafei:uaf-uafei-provider:0.2.09-SNAPSHOT 
  27.   +-com.uaf:microservice-apollo:0.2.09-SNAPSHOT 
  28.     +-com.google.inject:guice:4.1.0 
  29.       +-com.google.guava:guava:19.0 

结论

依赖冲突是开发过程中比较耗时的一个问题,通过诸如以上的一些功能协助我们尽快定位到问题,但最好的解决方式还是要遵循开发中的规范,约定优于配置。

 

责任编辑:姜华 来源: 编码是个技术活
相关推荐

2024-02-01 09:58:40

Java内存泄漏

2024-01-30 10:12:00

Java内存泄漏

2022-11-23 15:44:49

2022-12-04 23:54:39

2019-11-26 14:30:20

Spring循环依赖Java

2023-04-21 10:08:00

版本工具依赖关系

2023-05-04 08:06:27

Spring循环依赖

2020-11-04 10:11:22

区块链块链技术

2019-07-01 10:43:15

区块链技术数据

2018-03-12 13:25:51

2023-11-28 08:00:00

SpringJava

2024-09-12 15:36:57

2022-12-29 07:37:19

Java项目慢业务

2009-06-22 15:01:00

java项目常见错误

2023-12-21 08:51:37

防抖节流Vue.js

2019-03-25 10:30:19

开发技能代码

2024-11-06 09:44:22

2020-12-29 08:34:08

spring循环依赖开发

2020-10-27 14:15:42

SpringBoot

2020-06-03 10:09:03

微服务项目版本号
点赞
收藏

51CTO技术栈公众号