java程序员天天不是在建立jar包就是在建立jar包的路上,而且各类依赖引用都是以jar包的形式展现的。可是随着现代IDE的出现,我想不少程序员已经基本上不多直接和jar包打交道了。java
换句话说,他们已经不认识jar包了。程序员
那么jar包究竟是什么呢?它有哪些小秘密呢?一块儿来看一下吧。算法
jar包实际上是一种zip格式的文件,因此说你其实是可使用zip相关的命令来对jar包进行建立或者解压缩操做。apache
不一样的是jar包中多了一个META-INF文件夹。经过这个文件夹,jar包能够执行更多的操做。安全
JDK也自带了一个jar命令,经过jar命令咱们能够实现建立,更新jar包的操做,下图是JDK8中jar命令的说明:ide
由于JDK9以后引入了模块化的概念,因此JDK9以后jar命令有了比较大的变化:模块化
咱们看一下JDK14中的jar命令的用法:ui
这里主要不是讲jar命令,因此咱们不具体展开。spa
jar包和zip包最大的区别就在于jar包中包含了META-INF目录(不是必须的),咱们看一个比较经常使用的lombok.jar包的结构是怎么样的:code
这个版本比较新,因此它使用的是最新的JPMS的写法,你们能够看到在jar包的根目录下面有一个module-info.class文件,表示这个jar包使用的是模块化。
而后再看一下META-INF目录,里面有一个MANIFEST.MF文件:
Manifest-Version: 1.0 Ant-Version: Apache Ant 1.7.1 Created-By: 14.3-b01-101 (Apple Inc.) Premain-Class: lombok.launch.Agent Agent-Class: lombok.launch.Agent Can-Redefine-Classes: true Main-Class: lombok.launch.Main Lombok-Version: 1.18.10
MANIFEST.MF主要用来定义package相关的数据,这里咱们能够看到lombok的MANIFEST.MF文件定义了manifest的版本号,建立时间,版本号和几个类型的class。
services文件夹里面存放的能够对外提供的服务。
这里列出的文件并不全,实际上还有下面几种文件:
可使用 -i在生成jar包的时候自动建立,是class的index文件,主要用来加速class加载。
JAR包的签名文件。
与具备相同基本文件名的签名文件关联的签名块文件。 该文件存储相应签名文件的数字签名。
主要为使用多版本的特性准备的,里面存储的是不一样版本的class和资源。
好比下面命令建立了多个版本发行的jar包,而且将一些文件放在 META-INF/versions/9 目录中。
jar --create --file mr.jar -C foo classes --release 9 -C foo9 classes
假如咱们使用的是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模块化.
versions主要和 multi-release JAR一块儿使用的:
Multi-Release: true
所谓multi-release JAR就是说一个jar包能够支持不一样版本的JDK。咱们能够根据须要指定不一样版本的JDK所依赖的class文件或者属性文件。这个特性在咱们进行JDK升级的时候仍是颇有帮助的。
通常来讲,目录结构是这样的:META-INF/versions/N
其中N表示的是JDK的主要发行版本,好比9,10,11等。
类加载器会先去META-INF/versions/N目录中加载所须要的class,而后会去其余的低版本的META-INF/versions/N目录中加载所须要的class,最后才会在META-INF/的根目录加载其余的class文件。
MANIFEST.MF中存放的是key:value格式的配置信息,这些配置信息又能够分红两部分,第一部分是main-section信息,第二部分是individual-section。
咱们举个简单的例子:
Manifest-Version: 1.0 Created-By: 1.8 (Oracle Inc.) Sealed: true Name: foo/bar/ Sealed: false
其中
Manifest-Version: 1.0 Created-By: 1.8 (Oracle Inc.) Sealed: true
就是main-section信息,咱们用一张图来看一下main-section的信息有哪些:
在main-section信息下发能够接一个Name: Value,表示开启独立的针对于具体entry的属性(Per-Entry Attributes)配置:
Name: foo/bar/ Sealed: false
好比上面的属性是专门针对于包foo/bar/的,而且设置其Sealed属性为false。
Per-Entry Attributes除了 package versioning 和 sealing信息外,还能够定义Content-Type,Java-Bean,x-Digest-y和Magic属性。
JAR包能够经过使用jarsigner来对其进行签名。和签名相关的文件是:
签名事后的jar跟原来的jar其实并无什么不一样,只不过在META-INF/文件夹中多出了两个文件,一个是签名文件,一个是签名block文件。
签名文件是以.SF结尾的,这个文件和MANIFEST.MF很相似,能够指定Signature-Version和Created-By。
除此以外,还能够指定和安全相关的属性:
这两个属性主要用来作验证签名用的。
举个例子:
若是咱们的manifest是下面这样的:
Manifest-Version: 1.0 Created-By: 1.8.0 (Oracle Inc.) Name: common/class1.class SHA-256-Digest: (base64 representation of SHA-256 digest) Name: common/class2.class SHA1-Digest: (base64 representation of SHA1 digest) SHA-256-Digest: (base64 representation of SHA-256 digest)
那么相应的签名文件应该是这样的:
Signature-Version: 1.0 SHA-256-Digest-Manifest: (base64 representation of SHA-256 digest) SHA-256-Digest-Manifest-Main-Attributes: (base64 representation of SHA-256 digest) Name: common/class1.class SHA-256-Digest: (base64 representation of SHA-256 digest) Name: common/class2.class SHA-256-Digest: (base64 representation of SHA-256 digest)
若是再对.SF文件进行摘要,那么就会获得签名文件的摘要文件:
上面咱们讲到了一个Sealed属性:
Name: javax/servlet/internal/ Sealed: true
这个属性的意思是,javax/servlet/internal/包中的全部类必须从这个jar包中加载。
这个属性主要是从jar包的安全性来考虑的。
本文已收录于 http://www.flydean.com/java-jar-in-detail/最通俗的解读,最深入的干货,最简洁的教程,众多你不知道的小技巧等你来发现!
欢迎关注个人公众号:「程序那些事」,懂技术,更懂你!