Java打jar包的几种方式详解

一、制作只含有字节码文件的jar包

只含有字节码文件,即只含有class文件的jar包制作,这是最简单的形式

1、最简单的jar包——直接输出hello

最终生成的jar包结构
META-INF
Hello.class

方法步骤:
1️⃣用记事本写一个Hello.java的文件

class Hello{
  public static void main(String[] agrs){
   System.out.println("hello");
  }
}

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方法。

class Hello{
  public static void main(String[] agrs){
    Tom.speak();
  }
}
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。

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-INF\MENIFEST.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-INF\MENIFEST.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-INF\MENIFEST.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包。

三、制作含有资源文件的jar包

1、资源文件在jar包内部——读取jar内的文件

最终生成的jar包结构
META-INF
Hello.class
text.txt

方法步骤:

import java.io.InputStream;
import java.io.BufferedReader;
import java.io.InputStreamReader;

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一样,只不过需要在MENIFEST文件中将resource.jar加入classpath

import java.io.InputStream;
import java.io.BufferedReader;
import java.io.InputStreamReader;

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();
    }
}

3、资源文件在jar包外部——读取jar外的文件

最终生成的jar包结构
hello.jar
text.txt

方法步骤:

import java.io.InputStream;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.FileInputStream;

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();
    }
}

你可能感兴趣的:(Java打jar包的几种方式详解)