【问题解决】关于log4j与logback依赖冲突的解决方案

问题回放:

问题本质分析

  1. SLF4J 的角色:SLF4J 是一个日志门面(接口),它本身不记录日志,需要与一个具体的日志实现(如 Logback、Log4j2)绑定才能工作。

  2. "多个提供程序"警告:你的项目中同时存在两个 SLF4J 的具体实现(绑定器):

    ◦ org.apache.logging.slf4j.SLF4JServiceProvider -> 来自 Log4j2 的 log4j-slf4j2-impl 包。

    ◦ ch.qos.logback.classic.spi.LogbackServiceProvider -> 来自 Logback 框架。

  3. 致命错误:最关键的错误信息是:

    log4j-slf4j2-impl cannot be present with log4j-to-slf4j

    这意味着项目中同时包含了 log4j-slf4j2-impl 和 log4j-to-slf4j 这两个互斥的JAR包。

    ◦ log4j-slf4j2-impl:让 Log4j2 作为 SLF4J 的实现。

    ◦ log4j-to-slf4j:将其他的 Log4j2 日志调用桥接(重定向)到 SLF4J。如果 SLF4J 的背后又是 Log4j2,就会形成循环依赖,这是被禁止的。

结论

项目依赖混乱,同时引入了 Logback 和 Log4j2 两套完整的日志体系,并且 Log4j2 自身的两个模块也发生了冲突,导致 SLF4J 无法决定使用哪一个提供者,最终引发初始化异常。

解决方案

核心思路是:统一日志框架,只保留一个 SLF4J 绑定器。

你需要从 Log4j2 和 Logback 中二选一,并排除所有冲突的依赖。从日志看,系统最终选择的是 Log4j2 ([org.apache.logging.slf4j.SLF4JServiceProvider]),因此建议移除 Logback 和冲突的 Log4j2 桥接包。

方案一:选择使用 Log4j2

如果你的项目是 Spring Boot 应用,并且想使用 Log4j2,可以按以下步骤操作:

  1. 排除 Spring Boot 默认的 Logback 依赖。

    在 pom.xml 中,修改 spring-boot-starter 或 spring-boot-starter-web 的依赖:
    <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <!-- 排除默认的日志实现 Logback --> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-logging</artifactId> </exclusion> </exclusions> </dependency>

  2. 引入 Spring Boot 对 Log4j2 的官方支持 starter。
    <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-log4j2</artifactId> </dependency>

    这个 spring-boot-starter-log4j2 会自动为你引入正确且不冲突的 Log4j2 依赖(包括 log4j-core, log4j-api, log4j-slf4j2-impl),无需手动管理。

  3. 检查并移除其他可能引入冲突依赖的库。

    运行 mvn dependency:tree 命令,查看是否还有其他依赖引入了 log4j-to-slf4j 或 logback-classic。如果有,在相应的依赖项中添加 标签将其排除。

方案二:选择使用 Logback(更常见的 Spring Boot 默认方案)

如果你更倾向于使用 Spring Boot 默认的 Logback,则:

  1. 确保移除了所有 Log4j2 的核心依赖。
    在 pom.xml 中,查找并移除或排除 log4j-core, log4j-api, log4j-slf4j2-impl。
  2. 确保移除了 log4j-to-slf4j 桥接包(除非你有其他组件必须使用它来桥接)。
  3. 确保引入了 Logback 依赖(通常通过 spring-boot-starter-logging 传递引入,默认就有)。
  4. 移除或排除 spring-boot-starter-log4j2。

操作步骤总结(以 Maven 项目,选择方案一为例)

  1. 打开你的 pom.xml 文件。有可能存在多个,可以全局搜索。
  2. 在 spring-boot-starter-web 等核心 starter 中排除 spring-boot-starter-logging。
  3. 添加 spring-boot-starter-log4j2 依赖。
  4. 执行 Maven 命令更新依赖:
    mvn clean install
  5. 验证依赖树:
    mvn dependency:tree | grep -E "(log4j|logback|slf4j)"
    检查输出,确保只有 log4j-slf4j2-impl,而没有 logback-classic 和 log4j-to-slf4j。
  6. 重新启动你的应用 (RuoYiApplication),问题应该得到解决。

额外建议

• 理解依赖关系,请使用idea的mavenhelper插件

• 检查依赖传递:冲突常常由间接依赖引起。仔细分析 dependency:tree 的输出,找到是哪个顶级依赖带来了不需要的日志JAR包,然后对其进行排除。

相关推荐
TheRouter1 天前
LLM 应用的Evals 工程实践:从手动测试到自动化回归测试体系
运维·ai·自动化·log4j
老码观察2 天前
设计模式实战解读(二):工厂模式——对象创建的解耦艺术
设计模式·log4j
XiYang-DING3 天前
【Spring】SpringIoC&DI
java·spring·log4j
努力成为AK大王5 天前
超全 Maven 核心知识点总结
log4j
voyaqi8 天前
从零设计企业级校验框架:Spring Boot + SPI 实战指南
spring boot·后端·log4j
RemainderTime8 天前
(十二)Spring Cloud Alibaba 2023.x:基于 Filebeat 构建轻量级 ELK日志追踪体系
分布式·elk·elasticsearch·微服务·架构·logback
前端若水11 天前
智能体开发与传统软件开发的核心区别
网络·人工智能·python·开源·log4j
zlpzlpzyd13 天前
slf4j中jcl-over-slf4j、jul-to-slf4j、log4j-over-slf4j、slf4j-api的区别是什么
java·开发语言·log4j
计算机安禾14 天前
【c++面向对象编程】第9篇:友元(friend):破坏封装的“特权”——真的有害吗?
java·c++·log4j
代码漫谈15 天前
Spring Boot日志配置全攻略:打造高效、可靠的日志系统
java·spring boot·log4j·日志