Maven-优雅的替换第三方依赖中的类-《Java笔记》

admin 2025-10-19 01:11:19 编程 来源:ZONE.CI 全球网 0 阅读模式

Java Maven

1、背景

在项目中需要依赖其他第三方的jar包,但有时需要扩展第三方jar的功能,或者修复依赖中已知还未修复的bug,但碰到过很多种情况,第三方类库中并没有提供对应的扩展点导致无法优雅的使用继承等方法对代码进行改造。如果第三方类库并没有提供扩展点,通常会使用如下两种办法来进行二次开发:

  • 将第三方jar的源工程下载下来,放入maven中,重新进行修改,并发布在私有仓库中
  • 直接将需要修改的代码复制到当前工程,进行修改

如果这个第三方类库是全公司都需要使用的,使用第一种无疑是最好的,不过也带来了一定的维护成本。如果只是单个项目需要引入,直接将代码拷贝到工程中,直接修改,但编译后的文件会在当前classpath路径下生成对应的class文件,能否将这些修改的class打入到最终的第三方jar类库。

2、通过插件替换Jar包中的类

下面,介绍通过maven插件,将修改的类代码类,直接替换jar中的类。

2.1 maven-dependency-plugin

maven-dependency-plugin简单说明情况如下:

  • 插件说明:依赖项插件提供了操作工件的功能。它可以将项目从本地或远程存储库复制和/或解压缩到指定位置。官方地址
  • 使用命令:从存储库中检索项目列表并将其解压缩到定义位置的目标。具体命令如下:

    1. dependency:unpack
  • 操作原理

这个命令可以将指定的dependency解压到class目录中,然后设置不覆盖本地项目相同class文件(类的全限定名相同),就达到了本地文件替换源jar中class文件的目的。

2.2 实际操作

计划将org.apache.flink:flink-kubernetes_2.11:1.14.3中的KubernetesClusterClientFactory类,将createClusterDescriptor方法第一行输出日志。在需要的代码中添加一行日志如下图所示:

  1. package org.apache.flink.kubernetes;
  2. ...
  3. /** A {@link ClusterClientFactory} for a Kubernetes cluster. */
  4. @Internal
  5. public class KubernetesClusterClientFactory
  6. extends AbstractContainerizedClusterClientFactory<String> {
  7. ...
  8. @Override
  9. public KubernetesClusterDescriptor createClusterDescriptor(Configuration configuration) {
  10. //新添加代码
  11. System.out.println("configuration : "+ configuration)
  12. //=======
  13. checkNotNull(configuration);
  14. if (!configuration.contains(KubernetesConfigOptions.CLUSTER_ID)) {
  15. final String clusterId = generateClusterId();
  16. configuration.setString(KubernetesConfigOptions.CLUSTER_ID, clusterId);
  17. }
  18. return new KubernetesClusterDescriptor(
  19. configuration,
  20. FlinkKubeClientFactory.getInstance().fromConfiguration(configuration, "client"));
  21. }

为了在打包的时候修改的代码能打入到第三方jar包,需要修改pom文件,对应的代码如下:

  1. <properties>
  2. <flink.version>1.14.3</flink.version>
  3. <scala.binary.version>2.11</scala.binary.version>
  4. </properties>
  5. <dependencies>
  6. <dependency>
  7. <groupId>org.apache.flink</groupId>
  8. <artifactId>flink-kubernetes_${scala.binary.version}</artifactId>
  9. <version>${flink.version}</version>
  10. </dependency>
  11. </dependencies>
  12. <build>
  13. <plugins>
  14. <plugin>
  15. <groupId>org.apache.maven.plugins</groupId>
  16. <artifactId>maven-dependency-plugin</artifactId>
  17. <executions>
  18. <execution>
  19. <id>unpack</id>
  20. <phase>generate-sources</phase>
  21. <goals>
  22. <goal>unpack</goal>
  23. </goals>
  24. <configuration>
  25. <artifactItems>
  26. <artifactItem>
  27. <groupId>org.apache.flink</groupId>
  28. <artifactId>flink-kubernetes_${scala.binary.version}</artifactId>
  29. <overWrite>false</overWrite>
  30. <outputDirectory>${project.build.directory}/classes</outputDirectory>
  31. </artifactItem>
  32. </artifactItems>
  33. </configuration>
  34. </execution>
  35. </executions>
  36. </plugin>
  37. </plugins>
  38. </build>

接下来通过maven打包命令打包:

  1. mvm clean install

从class目录中查看KubernetesClusterClientFactory,发现代码已经是修改后的类。同时,当前项目编译的jar包,类也已经被替换成自定义的类,目标达成。

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  13