Dependency Conflict Resolution
About
A dependency conflict occurs in Maven when multiple dependencies in a project bring different versions of the same library (transitive dependencies), leading to:
Version Mismatches → Different versions of the same dependency in the classpath.
Compilation/Runtime Issues → Unexpected behavior due to incompatible versions.
ClassNotFoundException / NoSuchMethodError → When an older version is used instead of the expected one.
Maven resolves these conflicts automatically, but understanding the resolution mechanism helps avoid issues.
Maven does not allow multiple versions of the same library in the classpath simultaneously.
Why ?
Java’s classloading mechanism does not support multiple versions of the same class in the same classpath.
If two dependencies bring different versions of the same library, Maven picks one but warns about the conflict.
The other version is ignored, which can cause runtime errors if the selected version lacks required methods/classes.
How Do Dependency Conflicts Occur?
Consider a project with dependencies:
What Happens Internally?
spring-boot-starter-web
also depends onlog4j-core
, but at version 2.14.1.Now, two versions (2.17.1 & 2.14.1) exist in the dependency tree.
Maven must decide which version to use.
What Happens If Two Versions Are Declared?
Example: Conflicting Dependencies
Maven resolves this conflict automatically, choosing only one version based on:
Direct Dependency Preference – If one version is declared explicitly, it is preferred.
Nearest-Wins Strategy – If both are transitive dependencies, Maven picks the closest version.
To check which version is used:
Only one version appears in the resolved dependency tree.
Maven’s Conflict Resolution Strategy
1. Nearest-Wins Strategy (Dependency Mediation)
Maven chooses the nearest dependency version in the dependency tree.
If a dependency appears multiple times, the closest one to the project wins.
Example: Nearest-Wins in Action
If lib-B
internally depends on lib-A:1.0.0
, but the project declares lib-A:1.2.0
, Maven will use 1.2.0 because it is closer to the project.
Issue: This may cause incompatibilities if the newer version is not backward compatible.
2. Dependency Tree Analysis
Use the following command to check all dependencies and their versions:
Example output:
Solution: Manually enforce log4j-core:2.17.1
to resolve the conflict.
3. Forcing a Specific Version (Dependency Management)
If Maven picks an older or undesired version, explicitly override it using <dependencyManagement>
:
This ensures all modules in a multi-module project use the same version.
4. Using Exclusions to Remove Unwanted Dependencies
If a dependency is causing conflicts indirectly, we can exclude it:
This removes log4j-core
from spring-boot-starter-web
.
Be careful! Excluding dependencies may break functionality.
5. Using a BOM (Bill of Materials)
BOMs standardize versions across projects:
It ensures consistent versions across modules.
Can We Force Two Versions in the Same Project?
Not directly in the same classpath! But some workarounds exist:
Solution 1: Shading (Relocating Dependencies)
Used in fat JARs (e.g., with Maven Shade Plugin) to bundle different versions under separate namespaces.
Example: If two different libraries require different
jackson-databind
versions, we can relocate one version.
This moves jackson-databind:2.13.3
under shaded.com.fasterxml.jackson
, avoiding conflicts.
Solution 2: Isolating Dependencies in Different Classloaders
Used in OSGi, Java Modules (JPMS), or custom classloading strategies.
Example: Spring Boot’s parent-first and child-first classloading strategies.
Last updated
Was this helpful?