电脑故障

位置:IT落伍者 >> 电脑故障 >> 浏览文章

Jar文件包间接及Jar命令开发实例详解


发布日期:2018/11/28
 

常常看到有人询问如何把Java程序编译成exe文件通常回答只有两种一种是制作一个可执行的JAR文件包然后就可以像chm文档一样双击运行了而另一种是使用JET来进行编译

但是 JET 是要用钱买的而且据说 JET 也不是能把所有的 Java 程序都编译成执行文件性能也要打些折扣所以使用制作可执行 JAR 文件包的方法就是最佳选择了何况它还能保持 Java 的跨平台特性下面就来看看什么是 JAR 文件包吧

JAR 文件包

JAR 文件就是 Java Archive File顾名思意它的应用是与 Java 息息相关的是 Java 的一种文档格式JAR 文件非常类似 ZIP 文件准确的说它就是 ZIP 文件所以叫它文件包JAR 文件与 ZIP 文件唯一的区别就是在 JAR 文件的内容中包含了一个 METAINF/MANIFESTMF 文件这个文件是在生成 JAR 文件的时候自动创建的举个例子如果我们具有如下目录结构的一些文件

==

` test

` Testclass

把它压缩成 ZIP 文件 testzip则这个 ZIP 文件的内部目录结构为

testzip

` test

` Testclass

如果我们使用 JDK 的 jar 命令把它打成 JAR 文件包 testjar则这个 JAR 文件的内部目录结构为

testjar

| METAINF

| ` MANIFESTMF

` test

`Testclass

创建可执行的 JAR 文件包

制作一个可执行的 JAR 文件包来发布你的程序是 JAR 文件包最典型的用法

Java 程序是由若干个 class 文件组成的这些 class 文件必须根据它们所属的包不同而分级分目录存放运行前需要把所有用到的包的根目录指定给 CLASSPATH 环境变量或者 java 命令的 cp 参数运行时还要到控制台下去使用 java 命令来运行如果需要直接双击运行必须写 Windows 的批处理文件 (bat) 或者 Linux 的 Shell 程序因此许多人说Java 是一种方便开发者苦了用户的程序设计语言

其实不然如果开发者能够制作一个可执行的 JAR 文件包交给用户那么用户使用起来就方便了在 Windows 下安装 JRE (Java Runtime Environment) 的时候安装文件会将 jar 文件映射给 javawexe 打开

那么对于一个可执行的 JAR 文件包用户只需要双击它就可以运行程序了和阅读 chm 文档一样方便 (chm 文档默认是由 hhexe 打开的)那么现在的关键就是如何来创建这个可执行的 JAR 文件包

创建可执行的 JAR 文件包需要使用带 cvfm 参数的 jar 命令同样以上述 test 目录为例命令如下

jar cvfm testjar manifestmf test

这里 testjar 和 manifestmf 两个文件分别是对应的参数 f 和 m其重头戏在 manifestmf因为要创建可执行的 JAR 文件包光靠指定一个 manifestmf 文件是不够的因为 MANIFEST 是 JAR 文件包的特征可执行的 JAR 文件包和不可执行的 JAR 文件包都包含 MANIFEST关键在于可执行 JAR 文件包的 MANIFEST其内容包含了 MainClass 一项这在 MANIFEST 中书写格式如下

MainClass: 可执行主类全名(包含包名)

例如假设上例中的 Testclass 是属于 test 包的而且是可执行的类 (定义了 public static void main(String[]) 方法)那么这个 manifestmf 可以编辑如下

MainClass: testTest <回车>

这个 manifestmf 可以放在任何位置也可以是其它的文件名只需要有 MainClass: testTest 一行且该行以一个回车符结束即可创建了 manifestmf 文件之后我们的目录结构变为

==

| test

| ` Testclass

` manifestmf

这时候需要到 test 目录的上级目录中去使用 jar 命令来创建 JAR 文件包也就是在目录树中使用==表示的那个目录中使用如下命令

jar cvfm testjar manifestmf test

之后在==目录中创建了 testjar这个 testjar 就是执行的 JAR 文件包运行时只需要使用 java jar testjar 命令即可

需要注意的是创建的 JAR 文件包中需要包含完整的与 Java 程序的包结构对应的目录结构就像上例一样而 MainClass 指定的类也必须是完整的包含包路径的类名如上例的 testTest而且在没有打成 JAR 文件包之前可以使用 java <类名> 来运行这个类即在上例中 java testTest 是可以正确运行的 (当然要在 CLASSPATH 正确的情况下)

jar 命令详解

jar 是随 JDK 安装的在 JDK 安装目录下的 bin 目录中Windows 下文件名为 jarexeLinux 下文件名为 jar它的运行需要用到 JDK 安装目录下 lib 目录中的 toolsjar 文件不过我们除了安装 JDK 什么也不需要做因为 SUN 已经帮我们做好了我们甚至不需要将 toolsjar 放到 CLASSPATH 中

使用不带任何的 jar 命令我们可以看到 jar 命令的用法如下

jar {ctxu}[vfmM] [jar文件]

[manifest文件] [C 目录] 文件名

其中 {ctxu} 是 jar 命令的子命令每次 jar 命令只能包含 ctxu 中的一个它们分别表示

c创建新的 JAR 文件包

t列出 JAR 文件包的内容列表

x展开 JAR 文件包的指定文件或者所有文件

u更新已存在的 JAR 文件包 (添加文件到 JAR 文件包中)

[vfmM] 中的选项可以任选也可以不选它们是 jar 命令的选项参数

v生成详细报告并打印到标准输出

f指定 JAR 文件名通常这个参数是必须的

m指定需要包含的 MANIFEST 清单文件

只存储不压缩这样产生的 JAR 文件包会比不用该参数产生的体积大

但速度更快

M不产生所有项的清单(MANIFEST〕文件此参数会忽略 m 参数

[jar文件] 即需要生成查看更新或者解开的 JAR 文件包

它是 f 参数的附属参数

[manifest文件] 即 MANIFEST 清单文件它是 m 参数的附属参数

[C 目录] 表示转到指定目录下去执行这个 jar 命令的操作它相当于先使用 cd

命令转该目录下再执行不带 C 参数的 jar 命令它只能在创建和更新 JAR

文件包的时候可用

文件名 指定一个文件/目录列表这些文件/目录就是要添加到 JAR 文件包中的文件/目录如果指定了目录那么 jar 命令打包的时候会自动把该目录中的所有文件和子目录打入包中 下面举一些例子来说明 jar 命令的用法

) jar cf testjar test

该命令没有执行过程的显示执行结果是在当前目录生成了 testjar 文件如果当前目录已经存在 testjar那么该文件将被覆盖

) jar cvf testjar test

该命令与上例中的结果相同但是由于 v 参数的作用显示出了打包过程如下

标明清单(manifest)

增加test/(读入= )

(写出= )(存储了 %)

增加test/Testclass(读入= )

(写出= )(压缩了 %)

) jar cvfM testjar test

该命令与 ) 结果类似但在生成的 testjar 中没有包含 METAINF/MANIFEST 文件打包过程的信息也略有差别

增加test/(读入= )

(写出= )(存储了 %)

增加test/Testclass(读入= )

(写出= )(压缩了 %)

) jar cvfm testjar manifestmf test

运行结果与 ) 相似显示信息也相同只是生成 JAR 包中的 METAINF/MANIFEST 内容不同是包含了 manifestmf 的内容

) jar tf testjar

在 testjar 已经存在的情况下可以查看 testjar 中的内容如对于 ) 和 ) 生成的 testjar 分别应该此命令结果如下

对于 )

METAINF/

METAINF/MANIFESTMF

test/

test/Testclass

对于 )

test/

test/Testclass

) jar tvf testjar

除显示 ) 中显示的内容外还包括包内文件的详细信息

Wed Jun :: GMT METAINF/

Wed Jun :: GMT METAINF/MANIFESTMF

Wed Jun :: GMT test/

Wed Jun :: GMT test/Testclass

) jar xf testjar

解开 testjar 到当前目录不显示任何信息对于 ) 生成的 testjar解开后的目录结构如下

==

| METAINF

| ` MANIFEST

` test

`Testclass

jar xvf testjar

运行结果与 ) 相同对于解压过程有详细信息显示

创建METAINF/

展开METAINF/MANIFESTMF

创建test/

展开test/Test

上一篇:Notify()和notifyall()的原理

下一篇:Groovy编程用Closure替代匿名类