利用FatJar彻底解决Jar包冲突(一)

利用FatJar彻底解决Jar包冲突

  • FatJar的加载与隔离
    • ⼀、 FatJar概念
    • ⼆、FatJar的加载
    • 三、FatJar的隔离
    • 四、隔离机制验证
    • 五、 FatJar的定位
    • 六、 打包注意点

今天整理旧电脑里的资料,偶然翻到大概10年前实习时写的笔记,之前经常遇到Java依赖冲突的问题,想通过这种方式彻底解决Java里的依赖冲突,现在回过头来看,不知道是不是有点幼稚,欢迎交流,请轻喷。

FatJar的加载与隔离

⼀、 FatJar概念

将⼀个jar和他所依赖的jar都打在⼀个包中,这个包即为FatJar。

如何打FatJar

  1. 使⽤ maven shade 插件
    使⽤ maven shade 插件解压依赖的jar并和原⼯程class混在⼀起打包成⼀个jar。 优点:打包⽅式简单,之后加载也较容易。 缺点:⽬录多且乱,对于jar包中的配置⽂件不利于定位;对于内部依赖冲突这种打包⽅式会⾃动排除冲突,覆盖class⽂ 件,不利于排查jar包本身内部的冲突,如下图。
    建议:如果能够忍受这么乱的⽬录可以使⽤这种⽅式,因为加载class的时候很⽅便。
    使用maven shade 插件

  2. 使⽤ SpringBoot 提供的打包插件
    由于SpringBoot打出来的jar可以直接启动,这个jar就是FatJar,所以可以使⽤ Spring Boot 提供的打包插件将依赖的jar直接 打进FatJar,如下图。
    优点:通⽤,spring boot,pandora boot都是基于这种⽅式,很多问题都有现成的解决⽅案⽐如之后遇到的autoconfig注⼊问题,autoconfig 提供了针对fatjar注⼊的插件。⽬录⼲净明了,如下图。
    缺点:由于原⽣jar的加载只⽀持⼀层加载,即⽆法加载 jarin jar ⾥的class,所以这个问题需要调研,但是既然使⽤的是 Spring Boot的打包⽅式,Spring Boot本身肯定已经解决了这个问题。
    考虑到通⽤性与“美感”,下⾯针对这种打包⽅式来解决相应的问题。
    在这里插入图片描述

⼆、FatJar的加载

fatjar与普通jar的区别就是它将依赖的jar也打进了jar包⾥的lib⽬录下,所以需要加载jarin jar。对于Jar⾥的资源,定义 以‘!/ʼ来分割。原始的JarFile URL只⽀持⼀个‘!/ʼ,如图。
在这里插入图片描述
通过阅读Spring Boot启动加载相关的源码,发现Spring Boot扩展了这个协议,让它⽀持多个‘!/ʼ,从⽽就可以表示jarin jar的 资源了,如图。
在这里插入图片描述
既然这样,我们就可以复⽤Spring Boot 的 加载⽅式,即通过继承spring boot的launcher,复⽤它的createClassLoader来构造LaunchedURLClassLoader。
在这里插入图片描述

三、FatJar的隔离

解决了加载的问题之后,就需要研究如何与依赖FatJar的应⽤(以下简称为“应⽤”)隔离了,因为我们最终的⽬标就是针对 应⽤和jar⾥同类名(全路径)的两个class都能够加载且不冲突。根据java classloader的委托加载机制,我们可以确定我们 的FatJarClassLoader所处的位置如下:
在这里插入图片描述
应⽤的classloader使⽤的是AppClassLoader,需要创建⼀个与它同级的ClassLoader即FatJarClassLoader,通过设置 LaunchedClassLoader的parent为AppClassLoader的parent即ExtensionClassLoader就可以了,这样两个ClassLoader就可以 像上图那样隔离。
在这里插入图片描述

四、隔离机制验证

FatJarClassLoader 构造完成之后,我们就可以验证⼀下到底能不能解决冲突了。

  1. 原理层⾯验证
    验证代码如下:
    在这里插入图片描述
    输出结果如下:
    在这里插入图片描述
    可以看到 FatJarClassLoader 的 parent 为 ExtensionClassLoader,所以与AppClassLoader是同级的。String的ClassLoader为 空是由于BootStrapClassLoader是通过C++编写的,是最“底”层的ClassLoader。
  2. 实例验证-构造冲突
    在⼆⽅包中添加⽇志依赖如下:
    在这里插入图片描述
    该版本的Logger是有trace⽅法的,我们在⼆⽅包中可以直接调⽤进⾏确认:
    在这里插入图片描述
    将⼆⽅包打成普通的jar包,在应⽤中依赖该jar包(conflict-b),同时也依赖⼀个低版本的⽇志包:
    在这里插入图片描述
    在应⽤中调⽤⼆⽅包中的某个⽅法,该⽅法内部会调⽤Logger的trace⽅法:
    在这里插入图片描述
    报错如下:
    在这里插入图片描述
    可⻅构造冲突已经成功了!
  3. 实例验证-利⽤FatJar解决冲突
    将⼆⽅包打成FatJar之后在应⽤中通过反射调⽤原来会冲突的代码,调⽤成功:
    在这里插入图片描述
    在这里插入图片描述
    可⻅,⼆⽅包是通过隔离的⽅式加载的,加载的是⼆⽅包⾃带的⾼版本的Logger;⽽应⽤本身加载的还是⽼版本的 Logger。

五、 FatJar的定位

由于是在运⾏期加载FatJar,所以需要⼿动定位FatJar的位置,这边普通Java程序和Java Web 定位⽅式不同,⽬前只是同时 使⽤这两种⽅式进⾏加载,还没有其他更通⽤的解决⽅案。 当依赖⽅是普通 Java 程序时,所有依赖的jar包路径可以通过如下代码获得:
在这里插入图片描述
⽽当依赖⽅是Java Web 应⽤时,根据Tomcat 中应⽤的⽬录结构:
在这里插入图片描述
得到 jar 包路径的步骤为:
1. AppClassLoader.getResource(“/”).getPath()
2. 上溯并进⼊lib⽬录即可

六、 打包注意点

使⽤Spring Boot插件打出的FatJar是包含pom.xml 的,如下图:
在这里插入图片描述

可⻅pom⽂件位置和普通jar包的位置是⼀致的,这样的话应⽤maven其实是可以识别到FatJar的pom⽂件的,这样会将 FatJar依赖的jar包加⼊到应⽤⾥,还是需要⼿动排除冲突,这就跟我们的初衷不⼀致了。所以需要将pom⽂件在打包时排 除,如下代码:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/984775.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C/C++蓝桥杯算法真题打卡(Day4)

一、P11041 [蓝桥杯 2024 省 Java B] 报数游戏 - 洛谷 算法代码&#xff1a; #include<bits/stdc.h> using namespace std;// 计算第 n 个满足条件的数 long long findNthNumber(long long n) {long long low 1, high 1e18; // 二分查找范围while (low < high) {lo…

DeepSeek大语言模型下几个常用术语

昨天刷B站看到复旦赵斌老师说的一句话“科幻电影里在人脑中植入芯片或许在当下无法实现&#xff0c;但当下可以借助AI人工智能实现人类第二脑”&#xff08;大概是这个意思&#xff09; &#x1f49e;更多内容&#xff0c;可关注公众号“ 一名程序媛 ”&#xff0c;我们一起从 …

快速从C过度C++(一):namespace,C++的输入和输出,缺省参数,函数重载

&#x1f4dd;前言&#xff1a; 本文章适合有一定C语言编程基础的读者浏览&#xff0c;主要介绍从C语言到C过度&#xff0c;我们首先要掌握的一些基础知识&#xff0c;以便于我们快速进入C的学习&#xff0c;为后面的学习打下基础。 这篇文章的主要内容有&#xff1a; 1&#x…

IDEA接入阿里云百炼中免费的通义千问[2025版]

安装deepseek 上一篇文章IDEA安装deepseek最新教程2025中说明了怎么用idea安装codeGPT插件&#xff0c;并接入DeepSeek&#xff0c;无奈接入的官方api已经不能使用了&#xff0c;所以我们尝试从其他地方接入 阿里云百炼https://bailian.console.aliyun.com/ 阿里云百炼‌是阿…

六十天前端强化训练之第十三天之JavaScript 原型与继承详解

欢迎来到编程星辰海的博客讲解 目录 一、原型系统底层探秘 1.1 对象体系构建原理 内存模型示意图 1.2 原型链工作机制 1.3 原型相关方法详解 (1) Object.getPrototypeOf (2) Object.setPrototypeOf (3) Object.create 1.4 构造函数运行机制 二、八种继承模式深度剖析…

大语言模型学习--向量数据库基础知识

1.向量 向量是多维数据空间中的一个坐标点。 向量类型 图像向量 文本向量 语音向量 Embedding 非结构化数据转换为向量过程 通过深度学习训练&#xff0c;将真实世界离散数据&#xff0c;投影到高维数据空间上&#xff0c;通过数据在空间中间的距离体现真实世界的相似度 V…

Python语法核心架构与核心知识点:从理论到实践

一、Python的核心设计哲学 Python以“简洁优雅”为核心理念&#xff0c;遵循以下原则&#xff1a; # Zen of Python&#xff08;输入 import this 可查看&#xff09; >>> import this The Zen of Python, by Tim Peters ... Simple is better than complex. Readab…

mac本地部署Qwq-32b记录

导语 昨天看到阿里开源了Qwq-32b&#xff0c;号称性能可以媲美Deepseek-R1。今天晚上有空就在Mac上折腾了一下&#xff0c;使用ollma进行了部署&#xff0c;效果感觉还不错&#xff0c;特此记录。 环境 硬件 型号&#xff1a;Macbook M1 Pro 14寸内存&#xff1a;512G 环境…

Vue3 Pinia 符合直觉的Vue.js状态管理库

Pinia 符合直觉的Vue.js状态管理库 什么时候使用Pinia 当两个关系非常远的组件&#xff0c;要传递参数时使用Pinia组件的公共参数使用Pinia

JAVA面试_进阶部分_深入理解socket网络异常

在各种网络异常情况的背后&#xff0c;TCP是怎么处理的&#xff1f;又是怎样把处理结果反馈给上层应用的&#xff1f;本文就来讨论这个问题。 分为两个场景来讨论 建立连接 1 正常情况下 进过三次握手&#xff0c;客户端连接成功&#xff0c;服务端有一个新连接到来。 2 客…

警惕AI神话破灭:深度解析大模型缺陷与禁用场景指南

摘要 当前AI大模型虽展现强大能力&#xff0c;但其本质缺陷可能引发系统性风险。本文从认知鸿沟、数据困境、伦理雷区、技术瓶颈四大维度剖析大模型局限性&#xff0c;揭示医疗诊断、法律决策等8类禁用场景&#xff0c;提出可信AI建设框架与用户防护策略。通过理论分析与实操案…

《用Python+PyGame开发双人生存游戏!源码解析+完整开发思路分享》

导语​ "你是否想过用Python开发一款可玩性高的双人合作游戏&#xff1f;本文将分享如何从零开始实现一款类《吸血鬼幸存者》的生存射击游戏&#xff01;包含完整源码解析、角色系统设计、敌人AI逻辑等核心技术点&#xff0c;文末提供完整代码包下载&#xff01;" 哈…

Python实例:PyMuPDF实现PDF翻译,英文翻译为中文,并按段落创建中文PDF

基于PyMuPDF与百度翻译的PDF翻译处理系统开发:中文乱码解决方案与自动化排版实践 一 、功能预览:将英文翻译为中文后创建的PDF 二、完整代码 from reportlab.lib.pagesizes import letter from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle

华为hcia——Datacom实验指南——三层交换和ARP的工作原理

什么是三层交换 三层交换是指连接在同一台三层交换机上&#xff0c;不同vlan用户&#xff0c;不同网段ip&#xff0c;通过vlanif接口进行数据交换。 什么是ARP协议 通过网络层的ip地址解析成数据链路层的mac地址。 说白了就是通过目标ip地址去问他对应的mac地址是多少。 A…

YC 孵化项目 Pinch:实时语音翻译视频会议平台;Mistral OCR:能处理多语言多模态复杂文档丨日报

开发者朋友们大家好&#xff1a; 这里是 「RTE 开发者日报」 &#xff0c;每天和大家一起看新闻、聊八卦。 我们的社区编辑团队会整理分享 RTE&#xff08;Real-Time Engagement&#xff09; 领域内「有话题的 技术 」、「有亮点的 产品 」、「有思考的 文章 」、「有态度的 …

github生成badges的方法

在Github页面上生成类似下面这样的badge的方法 你可以通过以下步骤在GitHub个人主页的README中创建类似的技术栈徽章&#xff1a; 一、使用 Shields.io 生成徽章 Shields.io 是一个开源徽章生成工具&#xff0c;支持自定义文本、颜色、图标等参数。 1. 基础模板 https://…

Microsof Visual Studio Code 安装教程(中文设置)

VS Code 是一个免费的代码编辑器&#xff0c;可在 macOS、Linux 和 Windows作系统上运行。启动和运行 VS Code 既快速又简单。VS Code&#xff08;全称 Visual Studio Code&#xff09;是一款由Microsoft 推出的免费、开源、跨平台的代码编辑器&#xff0c;拥有强大的功能和灵活…

实验二:Git用法练习:使用 DevStar 搭建 Git 仓库托管平台

Git 是目前最流行的版本控制工具&#xff0c;而 GitHub、GitLab 等托管平台为团队协作提供了极大的便利。但在某些情况下&#xff0c;我们需要自建 Git 仓库托管平台&#xff0c;例如&#xff1a; 保护私有代码&#xff0c;避免托管到公网 提供企业或学校内部协作 提升对 Git…

一二三应用开发平台——能力扩展:多数据源支持

背景 随着项目规模的扩大&#xff0c;单一数据源已无法满足复杂业务需求&#xff0c;多数据源应运而生。 技术选型 MyBatis-Plus 的官网提供了两种多数据源扩展插件&#xff1a;开源生态的 <font style"color:rgb(53, 56, 65);">dynamic-datasource</fon…

【性能测试】Jmeter下载安装、环境配置-小白使用手册(1)

本篇文章主要包含Jmeter的下载安装、环境配置 添加线程组、结果树、HTTP请求、请求头设置。JSON提取器的使用&#xff0c;用户自定义变量 目录 一&#xff1a;引入 1&#xff1a;软件介绍 2&#xff1a;工作原理 3&#xff1a;安装Jmeter 4&#xff1a;启动方式 &#xf…