历史项目依赖库Bugfix技巧-类覆盖

在项目维护过程中,我们可能会遇到历史项目依赖的第三方库出现BUG而需要修复的情况,而这些第三方库可能来源于公司自主开发或开源项目,但由于各种原因,这些库可能已无人维护。

此时,解决这个问题有三个办法

一、基于源码修改重新打包发布

二、某些时候可能没有源码,需要反编译进行修改再重新编译打包

三、类覆盖

本文主要分享第三种类覆盖的实现方式,用于在一些特殊情况下解决线上问题,保证项目的正常运行。

一、类覆盖的基本原理

原理如其名,通过自定义类覆盖目标类。

具体实现方式为通过定义与三方库中存在问题目标类同名的自定义类,并确保其加载优先级高于原始类。

这里特别说明下类加载优先级的控制,分两种情况

1. 直接依赖

项目直接依赖了一个库C,项目源码打包后是在classes目录下的,此种情况无需特殊控制,默认classes目录优先加载,因此自定义的类会优先于依赖库中的类加载。

2. 关联依赖

项目依赖了库B,库B又依赖了库C,如果要控制库B和库C中同名类的加载顺序,可以通过定义依赖顺序进行控制。

以SpringBoot Maven 工程为例,B依赖定义在C依赖上面,可以保证最终生成的可执行jar中classpath.idx里面定义的库加载顺序B优先于C,保证最终加载时B库中的类优先于C库加载。

xml 复制代码
<dependency>
	<groupId>org.example</groupId>
	<artifactId>B</artifactId>
	<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
  <groupId>org.example</groupId>
  <artifactId>C</artifactId>
  <version>1.0-SNAPSHOT</version>
</dependency>

二、一个示例

如下工程,有一个依赖的三方库C

C依赖库种有一个C工具类,代码如下,其中的isEmpty方法未判断null值,会导致空指针异常

typescript 复制代码
package org.c;

public class C {

    public static boolean isEmpty(String s){
        return s.isEmpty();
    }

}

主模块A类代码如下,调用了C类的isEmpty方法

typescript 复制代码
public class A {
    public static void main(String[] args ) {
        String s = "123";

        if(!C.isEmpty(s)){
            System.out.println("String is " + s);
        }

        String ss = null;
        
        if(!C.isEmpty(ss)){ // 异常位置
            System.out.println("String is " + s);
        }

    }

}

运行A类将会在异常位置处出现空指针异常。

下面使用类覆盖方法进行BUG修正。

2.1 创建同名类

在A模块种创建同名类C,如下

自定义C类的代码如下

typescript 复制代码
package org.c;

public class C {

    public static boolean isEmpty(String s){
      // 修复原始库空指针问题
      if(s == null) {
         return true;
      }

        return s.isEmpty();
    }

}

2.2 验证效果

再次运行A类,程序正常执行完成。

三、总结

首先,优先推荐前两种方式进行问题修复,但是某些时候受制于时间、成本、风险等因素需要快速解决问题,这时,可以考虑第三种方案类覆盖。

需要特别注意的是,修改的方法可能会被多处调用,修改时需要充分考虑兼容性。

相关推荐
小毛驴8501 分钟前
spring-boot-maven-plugin,maven-compiler-plugin 功能对比
java·python·maven
csdn_aspnet19 分钟前
Java 霍尔分区算法(Hoare‘s Partition Algorithm)
java·开发语言·算法
霸道流氓气质25 分钟前
通义灵码 IDEA 插件完全使用指南
java·ide·intellij-idea
诸葛务农27 分钟前
道路行驶条件下电动汽车永磁电机的有效使用寿命及永磁体的失效和回收再利用(下)
java·开发语言·算法
Percep_gan38 分钟前
Java8中的stream的测试使用
java
砍材农夫1 小时前
物联网实战:Spring Boot MQTT | MQTT 设备模拟器演示(附源码)
java·spring boot·后端·物联网·spring·netty
我叫黑大帅1 小时前
解决聊天页内部滚轮改为页面滚动问题
javascript·后端·面试
EAIReport1 小时前
Spring AI 详解:Java 开发者快速落地 AI 应用
java·人工智能·spring
YDS8291 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— Agent执行链路设计之ReAct Loop
java·spring boot·ai·agent·deepseek
c++之路1 小时前
C++ 设计模式全总结
java·c++·设计模式