Maven生成可以直接运行的jar包的多种方式

Maven可以使用mvn package指令对项目进行打包,如果使用java -jar xxx.jar执行运行jar文件,会出现如下错误:

  • "no main manifest attribute, in xxx.jar"(没有设置Main-Class)
  • ClassNotFoundException(找不到依赖包)等错误。

要想jar包能直接通过java -jar xxx.jar运行,需要满足:

  1. 在jar包中的META-INF/MANIFEST.MF中指定Main-Class,这样才能确定程序的入口在哪里;
  2. 要能加载到依赖包。

使用Maven有以下几种方法可以生成能直接运行的jar包,可以根据需要选择一种合适的方法。

方法一:使用maven-jar-plugin和maven-dependency-plugin插件打包

pom配置

<build>
	<plugins>
 
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-jar-pluginartifactId>
			<version>2.6version>
			<configuration>
				<archive>
					<manifest>
						
						<addClasspath>trueaddClasspath>

						
						<classpathPrefix>lib/classpathPrefix>

						
						<mainClass>com.xxg.MainmainClass>
					manifest>
				archive>
			configuration>
		plugin>
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-dependency-pluginartifactId>
			<version>2.10version>
			<executions>
				<execution>
					<id>copy-dependenciesid>
					<phase>packagephase>
					<goals>
						<goal>copy-dependenciesgoal>
					goals>
					<configuration>
						
						<outputDirectory>${project.build.directory}/liboutputDirectory>
					configuration>
				execution>
			executions>
		plugin>
 
	plugins>
build>

MANIFEST.MF
maven-jar-plugin用于生成META-INF/MANIFEST.MF文件的部分内容.
下面是一个通过maven-jar-plugin插件生成的MANIFEST.MF文件片段:

Class-Path: lib/commons-logging-1.2.jar lib/commons-io-2.4.jar
Main-Class: com.xxg.Main

依赖jar
maven-dependency-plugin插件用于将依赖包拷贝到/lib目录下。
目录结构如下:
Maven生成可以直接运行的jar包的多种方式_第1张图片


指定了 Main-Class有了依赖包,那么就可以直接通过java -jar xxx.jar运行jar包。

这种方式生成jar包有个缺点,就是生成的jar包太多不便于管理,
下面两种方式只生成一个jar文件,包含项目本身的代码、资源以及所有的依赖包。

方法二:使用maven-assembly-plugin插件打包

pom配置

<build>
	<plugins>
 
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-assembly-pluginartifactId>
			<version>2.5.5version>
			<configuration>
				<archive>
					<manifest>
						<mainClass>com.xxg.MainmainClass>
					manifest>
				archive>
				<descriptorRefs>
					<descriptorRef>jar-with-dependenciesdescriptorRef>
				descriptorRefs>
			configuration>
		plugin>
 
	plugins>
build>

使用命令:mvn package assembly:single 打包即可。
打包后会在target目录下生成一个xxx-jar-with-dependencies.jar文件,这个文件不但包含了自己项目中的代码和资源,还包含了所有依赖包的内容。
所以可以直接通过java -jar来运行。

此外还可以直接通过mvn package来打包,无需assembly:single,不过需要加上一些配置:

<build>
	<plugins>
 
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-assembly-pluginartifactId>
			<version>2.5.5version>
			<configuration>
				<archive>
					<manifest>
						<mainClass>com.xxg.MainmainClass>
					manifest>
				archive>
				<descriptorRefs>
					<descriptorRef>jar-with-dependenciesdescriptorRef>
				descriptorRefs>
			configuration>
			<executions>
				<execution>
					<id>make-assemblyid>
					<phase>packagephase>
					<goals>
						<goal>singlegoal>
					goals>
				execution>
			executions>
		plugin>
 
	plugins>
build>

不过,如果项目中用到Spring Framework,用这种方式打出来的包运行时会出错,使用下面的方法三可以处理。

MAIN-添加任意key-value

<plugin>
  <groupId>org.apache.maven.pluginsgroupId>
    <artifactId>maven-assembly-pluginartifactId>
    <version>2.5.5version>
    <configuration>
      <archive>
        
        <manifestEntries>
          <Premain-Class>xx.yy.zz.PermainAgentPremain-Class>
        manifestEntries>
      archive>

    configuration>
   
  plugin>

方法三:使用maven-shade-plugin插件打包

pom配置

<build>
	<plugins>
 
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-shade-pluginartifactId>
			<version>2.4.1version>
			<executions>
				<execution>
					<phase>packagephase>
					<goals>
						<goal>shadegoal>
					goals>
					<configuration>
						<transformers>
							<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
								<mainClass>com.xxg.MainmainClass>
							transformer>
						transformers>
					configuration>
				execution>
			executions>
		plugin>
 
	plugins>
build>

配置完成后,执行mvn package即可打包。
在target目录下会生成两个jar包,注意不是original-xxx.jar文件,而是另外一个。

如果项目中用到了Spring Framework,将依赖打到一个jar包中,运行时会出现读取XML schema文件出错。原因是Spring Framework的多个jar包中包含相同的文件spring.handlers和spring.schemas如果生成一个jar包会互相覆盖
为了避免互相影响,可以使用AppendingTransformer来对文件内容追加合并

<build>
	<plugins>
 
		<plugin>
			<groupId>org.apache.maven.pluginsgroupId>
			<artifactId>maven-shade-pluginartifactId>
			<version>2.4.1version>
			<executions>
				<execution>
					<phase>packagephase>
					<goals>
						<goal>shadegoal>
					goals>
					<configuration>
						<transformers>
							<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
								<mainClass>com.xxg.MainmainClass>
							transformer>
							<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
								<resource>META-INF/spring.handlersresource>
							transformer>
							<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
								<resource>META-INF/spring.schemasresource>
							transformer>
						transformers>
					configuration>
				execution>
			executions>
		plugin>
 
	plugins>
build>

···

你可能感兴趣的:(Maven生成可以直接运行的jar包的多种方式)