java打jar包的几种方式详解-凯发平台
一、制作只含有字节码文件的jar包
我们先来看只含有字节码文件,即只含有class文件的jar包怎么制作,这是最简单的形式
1、最简单的jar包——直接输出hello
最终生成的jar包结构
meta-inf
hello.class
方法步骤
(1)用记事本写一个hello.java的文件
1 class hello{ 2 public static void main(string[] agrs){ 3 system.out.println("hello"); 4 } 5 }
(2)用命令行进入到该目录下,编译这个文件
javac hello.java
(3)将编译后的hello.class文件打成jar包
jar -cvf hello.jar hello.class
c表示要创建一个新的jar包,v表示创建的过程中在控制台输出创建过程的一些信息,f表示给生成的jar包命名
(4)运行jar包
java -jar hello.jar 这时会报如下错误 hello.jar中没有主清单属性
添加main-class属性
用压缩软件打开hello.jar,会发现里面多了一个meta-inf文件夹,里面有一个menifest.mf的文件,用记事本打开
1 manifest-version: 1.0 2 created-by: 1.8.0_121 (oracle corporation) 3
在第三行的位置写入 main-class: hello (注意冒号后面有一个空格,整个文件最后有一行空行),保存
再次运行 java -jar hello.jar ,此时成功在控制台看到 hello ,成功
2、含有两个类的jar包——通过调用输出hello
最终生成的jar包结构
meta-inf
tom.class
hello.class
方法步骤
(1)用记事本写一个hello.java和一个tom.java的文件
目的是让hello调用tom的speak方法
1 class hello{ 2 public static void main(string[] agrs){ 3 tom.speak(); 4 } 5 }
1 class tom{ 2 public static void speak(){ 3 system.out.println("hello"); 4 } 5 }
(2)编译: javac hello.java
此时hello.java和tom.java同时被编译,因为hello中调用了tom,在编译hello的过程中发现还需要编译tom
(3)打jar包,这次我们换一种方式直接定义main-class。
1 manifest-version: 1.0 2 created-by: 1.8.0_121 (oracle corporation) 3 main-class: hello 4
事先准备好上述的menifest.mf文件,并存放在meta-inf文件夹下,此时打jar包的命令如下
jar -cvfm hello.jar meta-infmenifest.mf hello.class tom.class
该命令表示用第一个文件当做menifest.mf文件,hello.jar作为名称,将hello.class和tom.class打成jar包。其中多了一个参数m,表示要定义menifest文件
(4)运行 java -jar hello.jar ,此时成功在控制台看到 hello ,成功
3、有目录结构的jar包——通过引包并调用输出hello
最终生成的jar包结构
meta-inf
com
tom.class
hello.class
我们将上一个稍稍变化一下,将tom这个类放在com包下,源文件目录结构变成
com
tom.java
hello.java
同时tom.java需要在第一行声明自己的包名
package com;
hello.java需要引入tom这个类,同样要在第一行进行import
import com.tom;
方法步骤
(1)编译hello.java
(2)打jar包,同样准备好menifest文件
jar -cvfm hello.jar meta-infmenifest.mf hello.class com
注意,最后一个com表示把com这个文件夹下的所有文件都打进jar包
(3)运行 java -jar hello.jar ,此时成功在控制台看到 hello ,成功
(4)优化过程
我们注意到,com包下是有tom.java源文件的,也被打进了jar包里,这样不太好,能不能优化一下javac命令,使所有的编译后文件编译到另一个隔离的地方呢,答案是可以的。
在编译hello.java时,先新建一个target文件夹。然后我们用如下命令
javac hello.java -d target
该命令表示,将所有编译后的文件,都放到target文件夹下。
将meta-inf文件夹也复制到target目录下,进入这个目录,输入如下命令
jar -cvfm hello.jar meta-infmenifest.mf *
注意最后一个位置变成了*,表示把当前目录下所有文件都打在jar包里
优化完毕
至此,我们可以总结出,制作一个只含有class字节码文件的jar包,以下命令足以
javac 要编译的文件 -d 目标位置
jar -cvfm 命名 menifest文件 要打包的文件1 要打包的文件2
二、制作含有jar文件的jar包
我们将场景稍稍变得复杂一点,看看jar包中需要引入其他jar包的场景
1、两个jar包间相互调用——调用jar外的jar输出hello
最终生成的jar包结构
hello.jar
tom.jar
方法步骤
准备:将上述一中写好的那个不带包的tom.jar复制过来(目的是调用里面的speak方法)
(1)编写一个hello.java并将其编译成hello.class,注意,由于hello里面引用了tom类的speak方法,因此在打jar包时应使用-cp参数,将tom.jar包引入
javac -cp tom.jar hello.class
这里的 -cp 表示 -classpath,指的是把tom.jar加入classpath路径下
(2)将hello.class达成jar包,步骤略
(3)此时运行 java -jar 发现报错 classnotfoundexception:tom
原因很简单,引入jar包需要在menifest.mf文件中配置一个新属性:class-path,路径指向你需要的所有jar包
现在menifest.mf这个文件应该变成
1 manifest-version: 1.0 2 created-by: 1.8.0_121 (oracle corporation) 3 main-class: hello 4 class-path: tom.jar 5
(4)好了,修改这个文件,再次运行,发现成功在控制台输出 hello
tips:引入多个jar包,中间用空格隔开
至此,我们可以总结出,命令变化如下
javac -cp xxx.jar 要编译的文件 -d 目标位置
jar -cvfm 命名 menifest文件 要打包的文件1 要打包的文件2
2、jar包中含有jar包——调用jar内的jar输出hello
最终生成的jar包结构
meta-inf
hello.class
tom.jar
当项目中我们把所需要的第三方jar包也打进了我们自己的jar包中时,如果仍然按照上述操作方式,会报找不到class异常。原因就是jar引用不到放在自己内部的jar包。
这种情况的具体实现细节比较复杂,我会在后一篇介绍一些知名的java应用是如何加载jar包的,来说明这种情况。实现方式的简单说明,可以先参考这篇文章:
http://www.cnblogs.com/adolfmc/archive/2012/10/07/2713562.html
三、制作含有资源文件的jar包
1、资源文件在jar包内部——读取jar内的文件
最终生成的jar包结构
meta-inf
hello.class
text.txt
方法步骤
1 import java.io.inputstream; 2 import java.io.bufferedreader; 3 import java.io.inputstreamreader; 4 5 class hello{ 6 public static void main(string[] args) throws exception{ 7 hello hello = new hello(); 8 inputstream is = hello.getclass().getresourceasstream("text.txt"); 9 print(is); 10 } 11 12 /** 13 * 读取文件,输出里面的内容,通用方法 14 */ 15 public static void print(inputstream inputstream) throws exception { 16 inputstreamreader reader = new inputstreamreader(inputstream, "utf-8"); 17 bufferedreader br = new bufferedreader(reader); 18 string s = ""; 19 while ((s = br.readline()) != null) 20 system.out.println(s); 21 inputstream.close(); 22 } 23 }
2、资源文件在另一个jar包内部——读取另一个jar内的文件
最终生成的jar包结构
hello.jar
resource.jar
text.txt
方法步骤
同1一样,只不过需要在menifest文件中将resource.jar加入classpath
1 import java.io.inputstream; 2 import java.io.bufferedreader; 3 import java.io.inputstreamreader; 4 5 class hello{ 6 public static void main(string[] args) throws exception{ 7 hello hello = new hello(); 8 inputstream is = hello.getclass().getresourceasstream("text.txt"); 9 print(is); 10 } 11 12 /** 13 * 读取文件,输出里面的内容,通用方法 14 */ 15 public static void print(inputstream inputstream) throws exception { 16 inputstreamreader reader = new inputstreamreader(inputstream, "utf-8"); 17 bufferedreader br = new bufferedreader(reader); 18 string s = ""; 19 while ((s = br.readline()) != null) 20 system.out.println(s); 21 inputstream.close(); 22 } 23 }
3、资源文件在jar包外部——读取jar外的文件
最终生成的jar包结构
hello.jar
text.txt
方法步骤
1 import java.io.inputstream; 2 import java.io.bufferedreader; 3 import java.io.inputstreamreader; 4 import java.io.fileinputstream; 5 6 class hello{ 7 public static void main(string[] args) throws exception{ 8 hello hello = new hello(); 9 inputstream is = new fileinputstream("text.txt"); 10 print(is); 11 } 12 13 /** 14 * 读取文件,输出里面的内容,通用方法 15 */ 16 public static void print(inputstream inputstream) throws exception { 17 inputstreamreader reader = new inputstreamreader(inputstream, "utf-8"); 18 bufferedreader br = new bufferedreader(reader); 19 string s = ""; 20 while ((s = br.readline()) != null) 21 system.out.println(s); 22 inputstream.close(); 23 } 24 }
出处:
https://www.cnblogs.com/mq0036/p/8566427.html