java打jar包的几种方式详解 |
您所在的位置:网站首页 › 现在支付有几种方式 › java打jar包的几种方式详解 |
本文对转载的原文章《java打jar包的几种方式详解》进行了部分内容的补充,添加了一些实际操作过程中的截图,方便大家更好的理解。 一、制作只含有字节码文件的jar包我们先来看只含有字节码文件,即只含有class文件的jar包怎么制作,这是最简单的形式 1、最简单的jar包——直接输出hello 最终生成的jar包结构 META-INF Hello.class方法步骤 (1)用记事本写一个Hello.java的文件,注意Java文件中不要带上所在的包package那一行代码 public class Hello { public static void main(String[] args) { System.out.println("hello"); } }(2)用命令行进入到该目录下,编译这个文件 javac Hello.java(3)将编译后的Hello.class文件打成jar包 jar -cvf hello.jar Hello.classc表示要创建一个新的jar包,v表示创建的过程中在控制台输出创建过程的一些信息,f表示给生成的jar包命名 (4)运行jar包 java -jar hello.jar这时会报如下错误 hello.jar中没有主清单属性 需要添加Main-Class属性 用压缩软件打开hello.jar,会发现里面多了一个META-INF文件夹,里面有一个MENIFEST.MF的文件,用Notepad++打开 在第三行的位置写入 Main-Class: Hello (注意冒号后面有一个空格,整个文件最后有一行空行),保存 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方法 public class Hello { public static void main(String[] args) { Tom.speak(); } } public class Tom { public static void speak(){ System.out.println("hello"); } }(2)编译: javac Hello.java 此时Hello.java和Tom.java同时被编译,因为Hello中调用了Tom,在编译Hello的过程中发现还需要编译Tom (3)打jar包,这次我们换一种方式直接定义Main-Class。 Manifest-Version: 1.0 Created-By: 1.7.0_80 (Oracle Corporation) Main-Class: Hello事先准备好上述的MENIFEST.MF文件,并存放在META-INF文件夹下,此时打jar包的命令如下 jar -cvfm hello.jar META-INF\MANIFEST.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;即 import com.Tom; public class Hello { public static void main(String[] args) { Tom.speak(); } } package com; public class Tom { public static void speak(){ System.out.println("hello"); } }方法步骤 (1)编译Hello.java (2)打jar包,同样准备好MENIFEST文件 事先准备好上述的MENIFEST.MF文件,并存放在META-INF文件夹下,此时打jar包的命令如下 注意,最后一个com表示把com这个文件夹下的所有文件都打进jar包 (3)运行 java -jar hello.jar ,此时成功在控制台看到 hello ,成功 (4)优化过程 我们注意到,com包下是有Tom.java源文件的,也被打进了jar包里,这样不太好,能不能优化一下javac命令,使所有的编译后文件编译到另一个隔离的地方呢,答案是可以的。 在编译Hello.java时,先新建一个target文件夹。然后我们用如下命令 该命令表示,将所有编译后的文件,都放到target文件夹下。 将META-INF文件夹也复制到target目录下,进入这个目录,输入如下命令 注意最后一个位置变成了*,表示把当前目录下所有文件都打在jar包里 优化完毕 至此,我们可以总结出,制作一个只含有class字节码文件的jar包,以下命令足以 javac 要编译的文件 -d 目标位置 jar -cvfm 命名 MENIFEST文件 要打包的文件1 要打包的文件2 二、制作含有jar文件的jar包 我们将场景稍稍变得复杂一点,看看jar包中需要引入其他jar包的场景 1、两个jar包间相互调用——调用jar外的jar输出hello 最终生成的jar包结构 hello.jar tom.jar Hello.java、Tom.java代码如下: public class Hello { public static void main(String[] args) { Tom.speak(); } } public class Tom { public static void speak(){ System.out.println("hello"); } }方法步骤 (1)javac Tom.java 编译Tom.java (2) jar -cvf tom.jar Tom.class 生成tom.jar (3)编写一个Hello.java并将其编译成Hello.class,注意,由于Hello里面引用了Tom类的speak方法,因此在打jar包时应使用-cp参数,将tom.jar包引入 javac -cp tom.jar Hello.java这里的 -cp 表示 -classpath,指的是把tom.jar加入classpath路径下 (4)将hello.class打成jar包 jar -cvfm hello.jar META-INF\MANIFEST.MF Hello.class(5)此时运行 java -jar 发现报错 ClassNotFoundException:Tom java -jar hello.jar原因很简单,引入jar包需要在MENIFEST.MF文件中配置一个新属性:Class-Path,路径指向你需要的所有jar包 现在MENIFEST.MF这个文件应该变成 (6)好了,修改这个文件后,重新打包hello.jar,再次运行,发现成功在控制台输出 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 方法步骤 import java.io.InputStream; import java.io.BufferedReader; import java.io.InputStreamReader; public class Hello{ public static void main(String[] args) throws Exception{ Hello hello = new Hello(); InputStream is = hello.getClass().getResourceAsStream("text.txt"); print(is); } public static void print(InputStream inputStream) throws Exception { InputStreamReader reader = new InputStreamReader(inputStream, "utf-8"); BufferedReader br = new BufferedReader(reader); String s = ""; while ((s = br.readLine()) != null) System.out.println(s); inputStream.close(); } }2、资源文件在另一个jar包内部——读取另一个jar内的文件 最终生成的jar包结构 hello.jar resource.jar text.txt 方法步骤 同1一样,只不过需要在MANIFEST文件中将resource.jar加入Class-Path 将text.txt生成resource.jar jar -cvf resource.jar text.txt其中,Hello.java的代码同1中 编译Hello.java javac -cp resource.jar Hello.java这里的 -cp 表示 -classpath,指的是把resource.jar加入classpath路径下 将Hello.class打成jar包 jar -cvfm hello.jar META-INF\MANIFEST.MF Hello.class运行java -jar hello.jar 3、资源文件在jar包外部——读取jar外的文件 最终生成的jar包结构 hello.jar text.txt 方法步骤 import java.io.InputStream; import java.io.BufferedReader; import java.io.InputStreamReader; public class Hello{ public static void main(String[] args) throws Exception{ Hello hello = new Hello(); InputStream is = new FileInputStream("text.txt"); print(is); } public static void print(InputStream inputStream) throws Exception { InputStreamReader reader = new InputStreamReader(inputStream, "utf-8"); BufferedReader br = new BufferedReader(reader); String s = ""; while ((s = br.readLine()) != null) System.out.println(s); inputStream.close(); } }
|
今日新闻 |
点击排行 |
|
推荐新闻 |
图片新闻 |
|
专题文章 |
CopyRight 2018-2019 实验室设备网 版权所有 win10的实时保护怎么永久关闭 |