JAVA程序员每天不是在创建jar包就是在创建jar包的路上,并且各种依赖引用都是以jar包的形式展示的 。但是随着现代IDE的出现,我想很多程序员已经基本上很少直接和jar包打交道了 。
换句话说,他们已经不认识jar包了 。
那么jar包到底是什么呢?它有哪些小秘密呢?一起来看一下吧 。
jar包到底是什么jar包其实是一种zip格式的文件,所以说你实际上是可以使用zip相关的命令来对jar包进行创建或者解压缩操作 。
不同的是jar包中多了一个META-INF文件夹 。通过这个文件夹,jar包可以执行更多的操作 。
JDK也自带了一个jar命令,通过jar命令我们可以实现创建,更新jar包的操作,下图是JDK8中jar命令的说明:
文章插图
因为JDK9之后引入了模块化的概念,所以JDK9之后jar命令有了比较大的变化:
我们看一下JDK14中的jar命令的用法:
文章插图
这里主要不是讲jar命令,所以我们不具体展开 。
META-INF目录jar包和zip包最大的区别就在于jar包中包含了META-INF目录(不是必须的),我们看一个比较常用的lombok.jar包的结构是怎么样的:
文章插图
这个版本比较新,所以它使用的是最新的JPMS的写法,大家可以看到在jar包的根目录下面有一个module-info.class文件,表示这个jar包使用的是模块化 。
然后再看一下META-INF目录,里面有一个MANIFEST.MF文件:
Manifest-Version: 1.0Ant-Version: Apache Ant 1.7.1Created-By: 14.3-b01-101 (Apple Inc.)Premain-Class: lombok.launch.AgentAgent-Class: lombok.launch.AgentCan-Redefine-Classes: trueMain-Class: lombok.launch.MainLombok-Version: 1.18.10
MANIFEST.MF主要用来定义package相关的数据,这里我们可以看到lombok的MANIFEST.MF文件定义了manifest的版本号,创建时间,版本号和几个类型的class 。services文件夹里面存放的可以对外提供的服务 。
这里列出的文件并不全,实际上还有下面几种文件:
- INDEX.LIST
- x.SF
- x.DSA
- versions/
比如下面命令创建了多个版本发行的jar包,并且将一些文件放在 META-INF/versions/9 目录中 。
jar --create --file mr.jar -C foo classes --release 9 -C foo9 classes
module-info.class假如我们使用的是JDK9之后的JPMS模块化,那么就会生成这么一个module-info.class 。这个文件主要是描述模块和外部模块直接的关系 。看一下lombok的例子:
module lombok {requires java.compiler;requires java.instrument;requires jdk.unsupported;requires static org.mapstruct.processor;exports lombok;exports lombok.experimental;exports lombok.extern.apachecommons;exports lombok.extern.java;exports lombok.extern.jbosslog;exports lombok.extern.log4j;exports lombok.extern.slf4j;exports lombok.extern.flogger;provides javax.annotation.processing.Processor with lombok.launch.AnnotationProcessorHider$AnnotationProcessor;provides org.mapstruct.ap.spi.AstModifyingAnnotationProcessor with lombok.launch.AnnotationProcessorHider$AstModificationNotifier;}
这里面我们定义了依赖的类和service providers,同时也定义了对外提供的类 。在JDK9之后,存在两种path,一种是之前的class path,一种是module path 。当 modular JAR被部署在module path中的时候,它就是一个modular JAR 。当他被部署在class path中的时候,就是一个non-modular JAR 。
同样的,如果是一个non-modular JAR被定义在module path中,那么这个non-modular JAR就自动被转换成了一个automatic module 。
如果jar包在MANIFEST.MF中定义了Automatic-Module-Name,那么module名字就是这个值,否则会从JAR的名字来定义这个module 。
automatic module主要是为了向下兼容而产生的 。
关于JPMS的更多信息可以参考我之前写的文章:JDK9的新特性:JPMS模块化.
versionsversions主要和 multi-release JAR一起使用的:
Multi-Release: true
所谓multi-release JAR就是说一个jar包可以支持不同版本的JDK 。我们可以根据需要指定不同版本的JDK所依赖的class文件或者属性文件 。这个特性在我们进行JDK升级的时候还是很有帮助的 。
推荐阅读
- Java9这些史诗级更新你都不知道?Java9新特性一文打尽
- Manjaro 和 Arch Linux 有何不同,哪个更好?
- 一文搞定 Koa 中间件实现原理
- 一文回顾深度学习发展史上最重要经典模型
- 五分钟让你读懂UML类图
- 一文让你读懂JAVA.IO、字符编码、URL和Spring.Resource
- 一文读懂神经网络的奥妙
- 一文看懂网上支付系统架构
- maven工程打包成jar 三国演义赵云什么时候投靠刘备
- 豪华车|WLTP续航660不缩水 奔驰最贵纯电车全球首发:一图读懂EQS SUV