使用Maven打包可运行jar和javaagent.jar的区别
简介
javaagent 是 Java1.5 之后引入的新特性,其主要作用是在class被加载之前对其拦截,以插入我们的字节码。
java1.5 之前使用的是JVMTI(jvm tool interface)技术来实现对class的拦截,不过这个是用 C++ 编写的,比如 debug 功能就是用这个技术实现的,有兴趣的自行百度。
jar
常见的jar包分为 可运行jar 和 javaagent.jar ,它们的主要区别如下:
| Executable Jar | Javaagent Jar | |
|---|---|---|
| 入口方法 | main | premain |
| 打包参数 | Main-Class | Premain-Class |
| 启动方式 | java -jar xxx.jar | -javaagent:xxx.jar=dddd |
| 是否可启动 | 是 | 否 |
入口方法
可运行 jar 包入口方法:
package org.coderead;
public class TestMain {
// 这个是我们经常写的
public static void main(String[] args) {
System.out.println("hello main");
}
}
javaagent.jar 入口方法:
package org.coderead;
import java.lang.instrument.Instrumentation;
public class TestAgent {
public static void premain(String arg, Instrumentation instrumentation) {
System.out.println("javaagent arg=" + arg);
}
}
MANIFEST.MF 文件方式
Use Your Own Manifest File 这篇官方文档介绍了使用 MANIFEST.MF 打包的方法:
<project>
...
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
...
<configuration>
<archive>
<manifestFile>src/main/resources/META-INF/MANIFEST.MF</manifestFile>
</archive>
</configuration>
...
</plugin>
</plugins>
</build>
...
</project>
默认情况下,Maven Archiver为您创建清单文件。有时使用您自己手工制作的清单文件很有用。假设您要使用清单文件 src/main/resources/META-INF/MANIFEST.MF。 通过将 <manifestFile> 配置元素设置为文件的位置来完成此操作。
您自己的清单文件的内容将与Maven Archiver创建的条目合并。 如果您在自己的清单文件中指定一个条目,它将覆盖Maven Archiver创建的值。
注意:与此处的所有示例一样,此配置可以在所有使用Maven Archiver的插件中使用,而不仅仅是本示例中的maven-jar-plugin。
- 打包一个可运行 jar 包 Executable.jar 的 MANIFEST.MF
Main-Class: org.coderead.TestMain
注意:
MANIFEST.MF 中需要多留一个空行,也就是输入完成最后一行之后回车
- 打包一个 javaagent 包 javaagent.jar 的 MANIFEST.MF
Premain-Class: org.coderead.TestAgent
Maven 项目结构
首先是 可运行jar 的项目结构:

然后是 javaagent.jar 的项目结构:

package 打包 jar

这个对应的命令是mvn package, 打包后会生成 target文件,并且生成 jar 包。
如果在 IDEA 没有找到右边栏,你可能需要点击IDEA左下角的按钮

验证
我在对两个项目执行了 package 之后,把 target\executable-1.0.jar 和 target\javaagent-1.0.jar 移动到同一个文件夹下,比如我在桌面新建了一个 test 文件夹。
然后执行命令
java -javaagent:javaagent-1.0.jar=test -jar executable-1.0.jar
执行结果如下图所示

manifestEntries 标签
还可以使用 <manifestEntries>代替创建 MANIFEST.MF 文件。这样就更加省事了!少创建一个文件!
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.coderead</groupId>
<artifactId>javaagent</artifactId>
<version>1.0</version>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<configuration>
<archive>
<!--方法一:MANIFEST.MF 配置文件-->
<!-- <manifestFile>src/main/resources/META-INF/MANIFEST.MF</manifestFile>-->
<!--方法二: pom 指定配置-->
<manifestEntries>
<Premain-Class>org.coderead.TestAgent</Premain-Class>
</manifestEntries>
</archive>
</configuration>
</plugin>
</plugins>
</build>
</project>
*manifest 标签
借助<manifest> 标签,我们可以指定可运行 jar 的主类。
参考官方文档 中关于<archive>标签讲解。
<archive>
<addMavenDescriptor/>
<compress/>
<forced/>
<index/>
<pomPropertiesFile/>
<manifestFile/>
<manifest>
<addClasspath/>
<addDefaultEntries/>
<addDefaultImplementationEntries/>
<addDefaultSpecificationEntries/>
<addBuildEnvironmentEntries/>
<addExtensions/>
<classpathLayoutType/>
<classpathPrefix/>
<customClasspathLayout/>
<mainClass/> <!--The Main-Class manifest entry.-->
<packageName/>
<useUniqueVersions/>
</manifest>
<manifestEntries>
<key>value</key>
</manifestEntries>
<manifestSections>
<manifestSection>
<name/>
<manifestEntries>
<key>value</key>
</manifestEntries>
<manifestSection/>
</manifestSections>
</archive>
因此,我们的代码还可以写成
<project>
...
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
...
<configuration>
<archive>
<!--指定 Main-Class 来指定可运行 jar 的主类 -->
<manifest>
<mainClass>org.coderead.TestMain</mainClass>
</manifest>
</archive>
</configuration>
...
</plugin>
</plugins>
</build>
...
</project>
总结
我们有两种方式来使用 Maven 打包 javaagent.jar 。
- 一种方式是在 <manifestFile> 标签中指定 META-INF/MANIFEST.MF 文件路径。在文件中填入 javaagent 打包属性。
- 另一种是在 <manifestEntries> 标签下,直接设置 javaagent 打包属性标签。
如果是打包可运行 jar 包,除了前两种方法外,第三种是使用 manifest 标签。
MANIFEST.MF 文件必须要在输入完最后一行配置后输入回车,即留出一行空白行
本文打包使用的插件是 maven-jar-plugin,生成 jar 包的命令是 mvn package
打包 javaagent.jar 必须指定 Premain-Class:org.coderead.TestAgent
使用Maven打包可运行jar和javaagent.jar的区别的更多相关文章
- maven打包 tomcat运行pom配置 或 打成jar包
maven打包 tomcat运行pom配置,同时还需要配置org.apache.tomcat.maven插件,这里省略. <groupId>com.company</groupId& ...
- 关于 maven 打包直接运行的 fat jar (uber jar) 时需要包含本地文件系统第三方 jar 文件的问题
关于maven打包fat jar (uber jar) 时需要包含本地文件系统第三方jar文件的问题,今天折腾了一整天.最后还是用了spring boot来做.下面是几篇关于打包的有参考价值的文章,以 ...
- idea在maven打包时运行Test测试, 导致打包失败, 乱七八糟的错误
在maven打包时运行Test测试, 导致打包失败, 乱七八糟的错误 在maven projects中图标toggle'skip Tests' Mode //宏杰帮助 网上案例:https://blo ...
- spring boot maven打包可运行jar包
普通打包之后在程序目录运行,或者编写bat运行时会提示“没有主清单属性”,这是因为并没有找到main()方法,需要我们指明告诉java程序 我bat中的代码 @echo off title mytit ...
- maven 打包可运行jar包(转)
目录 1.前提 2.方法一:使用maven-jar-plugin和maven-dependency-plugin插件打包 3.方法二:使用maven-assembly-plugin插件打包 4.方法三 ...
- 利用MAVEN打包可运行jar包,包括依赖的第三方包
转载自:http://bglmmz.iteye.com/blog/2058914 背景: 另一篇文章说了如何利用IDEA来打包,现在来说说如何利用MAVEN打包 目标:应用本身打成一个jar包,依赖的 ...
- 【Maven学习】Maven打包生成包含所有依赖的jar包
http://blog.csdn.net/u013177446/article/details/54134583 ******************************************* ...
- maven的几个重要配置文件pom.xml、settings.xml;Maven打包生成包含所有依赖的jar包
一个java项目通过maven自动下载依赖时,会涉级读取三个配置文件,分别是项目下的pom.xml 文件 .用户家目录下的.m2/settings.xml 与 maven 全局配置settings.x ...
- Maven 打包可运行 jar
为配合自动化部署hudson,最近研究了如何将eclipse maven工程打包成可运行的jar函数及对应的资源文件. 由于我们工程中包含了多个可运行的任务,在打包成jar时需要分别导出,p ...
随机推荐
- 【NX二次开发】属性操作相关函数的使用方法
内容包括:1.属性创建2.判断属性是否存在3.读取属性值4.时间属性转换成字符串5.统计属性的数量6.删除指定属性7.删除全部属性效果: 源码: #include <stdlib.h> # ...
- 【HTML】同页面锚点跳转
跳转: <a href="#maodian001">去吧!</a> 锚点: <a id="maodian001"></ ...
- ES6中的新数据类型——Symbol
今天小编和大家来聊一聊es6中新增的一个原始数据类型Symbol.在es5中原始数据类型(基本数据类型)有以下六种:Undefind.Null.Bool. String.Number.Object.今 ...
- ORA-09817:Write to audit file failed
ERROR: ORA-09817: Write to audit file failed. --这种错误基本是磁盘空间满了 问题描述:sqlplus / as sysdba登不上,报上述ora-098 ...
- 555定时器(1)单稳态触发器电路及Multisim实例仿真
555定时器(Timer)因内部有3个5K欧姆分压电阻而得名,是一种多用途的模数混合集成电路,它能方便地组成施密特触发器.单稳态触发器与多谐振荡器,而且成本低,性能可靠,在各种领域获得了广泛的应用. ...
- kubelet分析-csi driver注册源码分析
kubelet注册csi driver分析 kubelet注册csi driver的相关功能代码与kubelet的pluginManager有关,所以接下来对pluginManager进行分析.分析将 ...
- jquery循环动画
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title&g ...
- python基本函数增删改排序,用range()求和
a=["blue","red","brack"] print(len(a))#列表长度 a.append("yellow" ...
- Gitlab触发jenkins并获取项目post参数
jenkins -- Generic Webhook Trigger插件 此插件是git webhook的高阶应用,安装后会暴露出来一个公共API,GWT插件接收到 JSON 或 XML 的 HTTP ...
- css 字体阴影和边框阴影
字体阴影 text-shadow: 2px 2px 2px #7a7e81; 边框阴影 box-shadow: 2px 2px 2px #7a7e81;