命令行生成、运行及调用jar包
文章目录
- 一、制作只含有字节码文件的jar包
- 二、制作含有jar文件的jar包
- 三、制作含有资源文件的jar包
- 四、java命令行执行带依赖jar包的main函数
本文参考文章如下:
命令行生成并直接运行jar包
java命令行执行带依赖jar包的main函数
一、制作只含有字节码文件的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-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包。
这种情况的具体实现细节比较复杂,我会在后一篇介绍一些知名的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 }
四、java命令行执行带依赖jar包的main函数
有时候客户端没有运行环境,需要将程序放到服务器上执行,可按如下操作:
1、创建libs文件夹,将运行过程中依赖的jar包全部拷贝到此处;
2、编译、执行
1 javac -Djava.ext.dirs=libs MainApp.java
2 java -Djava.ext.dirs=libs MainApp
如果在MANIFEST.MF里配置了Main-Class,可以直接执行jar文件
java -Djava.ext.dirs=lib -jar xxx.jar
命令行生成、运行及调用jar包相关推荐
- java命令行生成jar_命令行生成可执行的jar包
jar包 jar包分类 jar包分为可执行jar包和不可执行jar包 jar包解压后都有一个META-INF/MANIFEST.MF文件 可执行jar包和不可执行jar包的区别是MANIFEST.MF ...
- java命令行执行程序解决依赖外部jar包的问题
java命令行执行程序解决依赖外部jar包的问题 参考文章: (1)java命令行执行程序解决依赖外部jar包的问题 (2)https://www.cnblogs.com/wangxiaoha/p/6 ...
- 命令行把java项目打成jar包
工作中需要把jenkin-client项目打成jar包使用. 以下为在命令行用jar命令打包,记录一下: 切换到项目路径,执行 jar cf jenkin-client.jar java-clie ...
- java 命令行 编译 jar文件_用命令行编译java并生成可执行的jar包
如果想用java编写一个可视化小程序,碰巧手头没有IDE的话,可以用命令行来完成编译.打包等工作. 拿自己编写的"java记事本"为例,介绍一下这个过程: 1.编写源代码. 编写源 ...
- Java按包编译jar_java用命令行编译运行时带有包和使用外部jar包的情况
命令行编译运行Java程序时,加载指定目录中的Jar包: 解决方案一: 编译:javac -Djava.ext.dirs=./lib Test.java 或 javac -Djava.ext.dir ...
- 用命令行编译java并生成可执行的jar包
1.编写源代码 编写源文件:CardLayoutDemo.java并保存,例如:I:\myApp\CardLayoutDemo.java.程序结构如下: package test;import jav ...
- 网络编程—使用C语言实现发送TCP数据包,以命令行形式运行:SendTCP source_ip source_port dest_ip dest_port;(原理和常见错误分析)
任务要求: 1.以命令行形式运行:SendTCP source_ip source_port dest_ip dest_port: 2.头部参数自行设定,数据字段为"This is my h ...
- Idea 中出现:运行 Test 时出错。命令行过长。 通过 JAR 清单或通过类路径文件缩短命令行,然后重新运行。
Idea 中出现:运行 Test 时出错.命令行过长. 通过 JAR 清单或通过类路径文件缩短命令行,然后重新运行. 在项目的 .idea 目录下的 workspace.xml 中的 <comp ...
- python运行代码不成功_命令行执行python模块时提示包找不到的问题
庄稼人不是专职python开发的道友,虽然与python相识已多年,可惜相识不相知,只是偶尔借助pydev写一些简单的小工具. 多年来,一直困惑于这样一个问题:同样的工程,同样的代码,使用pydev可 ...
最新文章
- OPUS 视频PPT介绍
- Linux配置免密登录
- html左侧分级导航,jquery实现无限分级横向导航菜单的方法
- Mysql分析排序和锁阅读总结
- 云计算是怎样入侵统一通信领域的
- php无符号整数转有符号整数,PHP中把有符号整型转换为无符号整型方法_PHP教程...
- 前端学习(2158):webpack配置文件的分离
- 思科交换机Portfast和Uplinkfast配置实验
- 关于状态更新时间字段取值的问题
- MS-SQL (sql server)添加数据库只读用户K3DBSQP,并遍历所有数据库添加访问只读访问权限
- python vimIDE环境
- 中国最后的江湖客:有3000万人,叫卡车司机
- 2021-09-01 myAtoi(
- 程序员该如何规划自己的职业发展?
- 2022年中国中小学教育信息化行业研究报告
- Namesilo转出域名到US Domain Center美国域名注册商
- 效率神器----WOX
- 文件包含漏洞及简单绕过
- About MyBatis——有关Mybatis,自问自答系列
- 十五、Fluent湿空气模拟-组分输运模型
热门文章
- ipynb转py命令
- springboot动态添加log4j2的Appender
- MAC地址 单播,组播,广播
- 数据仓库、主题域、主题概念与定义
- 【Python 百练成钢】高精度加法、阶乘计算、矩阵幂运算、矩阵面积交
- 【模电】0016 线性稳压电源的几个实用电路
- 基于stm32单片机的信号发生器设计
- C++ 强制类型转换:static_cast、dynamic_cast、const_cast 和 reinterpret_cast
- 我在linux主机用arm-linux-gcc编译后,复制到开发板根目录运行,出现/bin/sh: ./hello not found
- Entity Framework Core-DbContext