jar命令(一)-jar打包class文件
2017-03-16 17:44
162 查看
jar包本质上是将所有class文件、资源文件压缩打成一个包(也可以选择不压缩),可选择在jar包中生成META-INF/MANIFEST.MF文件,MANIFEST.MF是清单文件,里面可以记录主类、classpath等信息,供虚拟机使用。
接下来的一段时间里,我们将以以下路径学习jar命令和清单文件的相关知识
jar打包class文件
带包class文件jar打包
清单文件的使用
在这篇文章里我们将使用简单的java程序来熟悉jar命令的使用,因为是出于熟练使用jar的目的,下面的操作中可能会啰里啰嗦、重复使用jar命令,以下是本片文章的目录,精简版直接看 jar命令的选项
一 jar命令的选项
用法
选项
本节中使用到的命令
二 只有一个class文件的可执行jar的实现
方法一将Mainjar全部解压修改MANIFESTMF文件后再重新打包
方法二生成jar包的时候为其指定清单文件
方法三为jar包指定清单文件
方法四为jar更新文件文件是清单文件
三 多个class的打包
在Main.java里面添加我们的hello world代码,并javac编译
然后将生成的Main.class打包,c选项是创建一个新的jar包,f是指定jar包的名字
如果附加使用v选项,则会列出详细的打包信息
我们可以通过t选项来查看生成的jar中的文件,输出的内容也就是包目录和包文件,可以看到是默认生成MANIFEST.MF文件的
当然我们也可以通过M选项不生成MANIFEST.MF文件,可以看到包中只有Main.class一个文件
虽然我们已经将Main.class打包了,并且Main.class有main(String … args)入口方法,可是这个jar还是不能执行,因为虚拟机并不知道这个包中的哪个class中是有main方法的
下面还是恢复到有MANIFEST.MF文件的Main.jar包,我们要将他解压,在MANIFEST.MF中添加主类属性
解压jar包的方法有很多种,我们解压的目的是为了查看默认生成的MANIFEST.MF文件,并做修改,最后生成一个可执行的jar包,因为我们的目的是出于熟练使用jar命名,下面我们将使用多个方法实现这个目标
打开META-INF/MANIFEST.MF文件,看到默认生成的清单文件很简单,只有两行
添加上添加上主类属性,注意冒号后面一定要留一个空格,最后一定要多留一个换行符
然后使用再将uncompress中所有目录、文件全部打包,注意先将原来的Main.jar删除,最后就可以运行这个新的有主类属性的Main.jar啦
注意上面这个jar命令使用的M选项是不生成清单,因为原本uncompress目录里就有,如果还是使用jar cf命令,生成的清单和上面第一次打包生成清单的是一样的,没有Main-Class: Main主类
接下了指定清单文件,生成Main.jar,m选项可以指定一个清单文件
jardemo/uncompress目录下的的MANIFEST.MF是有主类的,我们把它移到jardemo目录下
使用u选项更新jar包,使用m选项指定清单文件,输出了一些警告信息,因为新的清单文件和Main.jar包中原来的清单信息有两个字段是重名的
再运行这个jar
我们在jardemo目录里删除有主类清单的Main.jar,再次生成最早的MANIFEST.MF中没有主类的Main.jar
jar包中的MANIFEST.MF里记录了主类,而MANIFEST.MF只是jar包中的一个文件,我们将Main.jar中没有主类的MANIFEST.MF替换为有主类的MANIFEST.MF和方法三是同样的效果,首先要讲有主类的MANIFEST.MF放在合适的路径里面,我们之前看到jar包中的文件是放在META-INF目录里的,使用u来更新jar,注意需要使用M来忽视清单文件
在新建的Say.java:
修改Main.java文件,让它调用Say类中的Say.say(…)方法
我们编译文件并打包,注意指定一个那个有主类属性的MANIFEST.MF
接下来的一段时间里,我们将以以下路径学习jar命令和清单文件的相关知识
jar打包class文件
带包class文件jar打包
清单文件的使用
在这篇文章里我们将使用简单的java程序来熟悉jar命令的使用,因为是出于熟练使用jar的目的,下面的操作中可能会啰里啰嗦、重复使用jar命令,以下是本片文章的目录,精简版直接看 jar命令的选项
一 jar命令的选项
用法
选项
本节中使用到的命令
二 只有一个class文件的可执行jar的实现
方法一将Mainjar全部解压修改MANIFESTMF文件后再重新打包
方法二生成jar包的时候为其指定清单文件
方法三为jar包指定清单文件
方法四为jar更新文件文件是清单文件
三 多个class的打包
一. jar命令的选项
用法:
大括号中的选项是必选的,中括号里选项是可选的,jar-file是jar文件;manifest-file是清单文件,即jar包中的META-INF/MANIFEST.MF文件jar {ctxui}[vfmn0PMe] [jar-file] [manifest-file] [entry-point] [-C dir] files ...
选项:
-c (create)创建新档案 -t 列出档案目录 -x 从档案中提取指定的 (或所有) 文件 -u (update)更新现有档案 -v 在标准输出中生成详细输出 -f 指定档案文件名 -m 包含指定清单文件中的清单信息 -n 创建新档案后执行 Pack200 规范化 -e 为捆绑到可执行 jar 文件的独立应用程序 指定应用程序入口点 -0 仅存储; 不使用任何 ZIP 压缩 -P 保留文件名中的前导 '/' (绝对路径) 和 ".." (父目录) 组件 -M 不创建条目的清单文件 -i 为指定的 jar 文件生成索引信息 -C 更改为指定的目录并包含以下文件
本节中使用到的命令
jar uf xx.jar [file … | path]是需要注意的,它在更新完jar文件后会生成新的清单文件,这一点在本篇文章的 “方法4:为jar更新文件,文件是清单文件” 中有实例说明jar cf xx.jar [file ... | path] 将file等文件或path目录打包成xx.jar jar cvf xx.jar [file ... | path] 同上,显示详细信息 jar cmf manifest-file xx.jar [file ... | path] 将file等文件或path目录打包到xx.jar,并制定它的清单文件 jar cMf xx.jar [file ... | path] 将file等文件或path目录打包到xx.jar,包中不生成清单文件 jar uf xx.jar [file ... | path] 将file等文件或目录更新到xx.jar,务必注意!!!这个更新会重新生成清单文件 jar uMf xx.jar [file ... | path] 同上,但不会生成清单文件 jar umf manifest-file xx.jar {file ... | path} 将file等文件或path目录更新到xx.jar jar tf xx.jar 列出xx.jar中所有文件 jar xf xx.jar 把xx.jar中所有文件提取到当前目录 jar xf xx.jar {file ...} 把xx.jar中file等文件提取到当前目录
二. 只有一个class文件的可执行jar的实现
首先创建一个jarDemo文件夹,里面放置我们的测试类mkdir jarDemo cd jarDemo touch Main.java
在Main.java里面添加我们的hello world代码,并javac编译
public class Main{ public static void main(String ... args){ System.out.println("hello world"); } }
javac Main.java
然后将生成的Main.class打包,c选项是创建一个新的jar包,f是指定jar包的名字
jar cf Main.jar Main.class
如果附加使用v选项,则会列出详细的打包信息
jar cvf Main.jar Main.class
我们可以通过t选项来查看生成的jar中的文件,输出的内容也就是包目录和包文件,可以看到是默认生成MANIFEST.MF文件的
jar tf Main.jar 输出: META-INF/ META-INF/MANIFEST.MF Main.class
当然我们也可以通过M选项不生成MANIFEST.MF文件,可以看到包中只有Main.class一个文件
rm Main.jar jar cMf Main.jar Main.class jar tf Main.jar 输出: Main.class
虽然我们已经将Main.class打包了,并且Main.class有main(String … args)入口方法,可是这个jar还是不能执行,因为虚拟机并不知道这个包中的哪个class中是有main方法的
java -jar Main.jar 输出: Main.jar中没有主清单属性
下面还是恢复到有MANIFEST.MF文件的Main.jar包,我们要将他解压,在MANIFEST.MF中添加主类属性
解压jar包的方法有很多种,我们解压的目的是为了查看默认生成的MANIFEST.MF文件,并做修改,最后生成一个可执行的jar包,因为我们的目的是出于熟练使用jar命名,下面我们将使用多个方法实现这个目标
方法一:将Main.jar全部解压,修改MANIFEST.MF文件后再重新打包
因为jar解压命令只能解压到当前目录,这样会造成文件混乱,我们新建一个uncompress文件夹,将Main.jar拷贝到这个文件夹后再解压,可以看到,jar包中所有文件,x选项是提出全部或指定文件,这个例子里是提出全部文件,实际中可以用f选项提出指定文件,比如 >> jar xf Main.jar META-INF/MANIFEST.MFmkdir uncompress cp Main.jar uncompress cd uncompress jar xf Main.jar ls -RF 输出: META-INF/ Main.class Main.jar ./META-INF: MANIFEST.MF
打开META-INF/MANIFEST.MF文件,看到默认生成的清单文件很简单,只有两行
Manifest-Version: 1.0 Created-By: 1.8.0_101 (Oracle Corporation)
添加上添加上主类属性,注意冒号后面一定要留一个空格,最后一定要多留一个换行符
Manifest-Version: 1.0 Created-By: 1.8.0_101 (Oracle Corporation)Main-Class: Main
然后使用再将uncompress中所有目录、文件全部打包,注意先将原来的Main.jar删除,最后就可以运行这个新的有主类属性的Main.jar啦
rm Main.jar jar cMf Main.jar . java -jar Main.jar 输出: hello world
注意上面这个jar命令使用的M选项是不生成清单,因为原本uncompress目录里就有,如果还是使用jar cf命令,生成的清单和上面第一次打包生成清单的是一样的,没有Main-Class: Main主类
rm Main.jar
jar cf Main.jar .
java -jar Main.jar 输出: Main.jar中没有主清单属性
方法二:生成jar包的时候为其指定清单文件
还是那个方法一的uncompress目录,我们把META-INF下的MANIFEST.MF移到uncompress目录下,删除Main.jar,和META-INF目录,让uncompress中只有Main.class和MANIFEST.MF两个文件,MANIFEST.MF是有主类属性的清单文件mv META-INF/MANIFEST . rm Main.jar rm -r META-INF ls -RF 输出: MANIFEST.MF Main.class
接下了指定清单文件,生成Main.jar,m选项可以指定一个清单文件
jar cmf MANIFEST.MF Main.jar Main.class java -jar Main.jar 输出: hello world
方法三:为jar包指定清单文件
还记得最早那个jardemo目录里的那个清单里没有主类的Main.jar吗,我们为它更新有主类的清单文件jardemo/uncompress目录下的的MANIFEST.MF是有主类的,我们把它移到jardemo目录下
cd desktop/jardemo mv uncompress/MANIFEST.MF . rm -r uncompress ls -RF 输出: MANIFEST.MF Main.class Main.jar Main.java
使用u选项更新jar包,使用m选项指定清单文件,输出了一些警告信息,因为新的清单文件和Main.jar包中原来的清单信息有两个字段是重名的
jar -umf MANIFEST.MF Main.jar 输出: 三月 13, 2017 10:50:50 下午 java.util.jar.Attributes read WARNING: Duplicate name in Manifest: Manifest-Version. Ensure that the manifest does not have duplicate entries, and that blank lines separate individual sections in both your manifest and in the META-INF/MANIFEST.MF entry in the jar file. 三月 13, 2017 10:50:50 下午 java.util.jar.Attributes read WARNING: Duplicate name in Manifest: Created-By. Ensure that the manifest does not have duplicate entries, and that blank lines separate individual sections in both your manifest and in the META-INF/MANIFEST.MF entry in the jar file.
再运行这个jar
java -jar Main.jar 输出: hello world
方法四:为jar更新文件,文件是清单文件
使用这个方法主要是为了再次熟悉一下u选项和理解包的概念我们在jardemo目录里删除有主类清单的Main.jar,再次生成最早的MANIFEST.MF中没有主类的Main.jar
jar cf Main.jar Main.class
jar包中的MANIFEST.MF里记录了主类,而MANIFEST.MF只是jar包中的一个文件,我们将Main.jar中没有主类的MANIFEST.MF替换为有主类的MANIFEST.MF和方法三是同样的效果,首先要讲有主类的MANIFEST.MF放在合适的路径里面,我们之前看到jar包中的文件是放在META-INF目录里的,使用u来更新jar,注意需要使用M来忽视清单文件
mkdir META-INF
mv MANIFEST.MF META-INF
jar uMf Main.jar META-INF/MANIFEST.MF
java -jar Main.jar 输出: hello world
三. 多个class的打包
上一节中,jarDemo目录中有Main.java和有主类信息的清单文件MANIFEST.MF,我们只保留这两个个文件,其他全部删掉,然后新建一个Say.java文件ls -RF 输出: META-INF/ Main.java Say.java ./META-INF: MANIFEST.MF
在新建的Say.java:
public class Say{ public static void say(String str){ System.out.println("This is " + str); } }
修改Main.java文件,让它调用Say类中的Say.say(…)方法
public class Main{ public static void main(String ... args){ System.out.println("hello world"); Say.say("Charles"); } }
我们编译文件并打包,注意指定一个那个有主类属性的MANIFEST.MF
javac *.java
jar cmf Main.jar META-INF/MANIFEST.MF Main.class Say.class
java -jar Main.jar 输出: hello world
This is Charles
相关文章推荐
- 编译class文件与jar打包命令--Java基础056
- 用jar命令将java class文件打包以及编写方便执行jar文件的批处理文件
- 把java中的class文件打包成.jar (jar命令详解)
- java 自带jar命令打包class文件
- java 自带jar命令打包class文件
- java的*.class打包成jar文件
- DOS命令 SHELL命令 BAT命令 执行 java 的 class文件和Jar文件
- 将Java *.Class 打包成可执行的*.jar文件
- 把class文件打成jar包的命令
- 将.class文件打包成jar文件
- swing中通过.class文件(把.java都删除)打包成jar文件,使用皮肤包遇到的问题。
- jar 打包java class文件后,运行jar 出现 Failed to load Main-Class manifest attribute from 错误
- java中jar命令打包一个文件夹下的所有文件
- java .class文件打包成.jar文件
- 将eclipse下编译的class文件打包成hadoop集群可以运行的jar包的过程
- 把java文件打包成.jar (jar命令详解)
- java中.class文件打包成.jar文件问题
- jdk自带工具 如何把class 文件 打成jar包 命令
- [原创]用jar命令将Web应用打包成war文件的简单方法
- Java打包指南-JAR文件包及jar命令详解