Maven打包方式






1、背景描述


Maven可以使用mvn package指令对项目进行打包,如果使用java -jar执行jar文件,会出现没有设置Main-Class、ClassNotFoundException(找不到依赖包)等错误

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

  • jar包中的META-INF/MANIFEST.MF中指定Main-Class,以确定程序入口
  • 要能加载到jar包中所依赖的其它依赖包

使用Maven有以下几种方式可以生成能直接运行的jar包:

2、使用maven-jar-plugin和maven-dependency-plugin插件打包


maven-jar-pluginmaven-dependency-plugin是默认的打包方式,用来将普通的Project打成jar

<build>
	<plugins>
		<plugin>
			<groupId>org.apache.maven.plugins</groupId>
			<artifactId>maven-jar-plugin</artifactId>
			<version>2.6</version>
			<configuration>
				<archive>
					<manifest>
					    <!-- 是否添加依赖的jar路径配置 -->
						<addClasspath>true</addClasspath>
						<!-- 依赖的jar包存放未知,和生成的jar放在同一级目录下 -->
						<classpathPrefix>lib/</classpathPrefix>
						<!-- 指定入口类 -->
						<mainClass>com.cc.Main</mainClass>
					</manifest>
				</archive>
				<!-- 不打包com.cc.excludes下面的所有类 -->
                <excludes>com/cc/excludes/*</excludes>
			</configuration>
		</plugin>
		<plugin>
			<groupId>org.apache.maven.plugins</groupId>
			<artifactId>maven-dependency-plugin</artifactId>
			<version>2.10</version>
			<executions>
				<execution>
					<id>copy-dependencies</id>
					<phase>package</phase>
					<goals>
						<goal>copy-dependencies</goal>
					</goals>
					<configuration>
						<outputDirectory>${project.build.directory}/lib</outputDirectory>
					</configuration>
				</execution>
			</executions>
		</plugin>
	</plugins>
</build>

可以使用includesexcludes选择只打包的部分内容

maven-jar-plugin用于生成META-INF/MANIFEST.MF文件的部分内容:

  • <mainClass>com.cc.Main</mainClass>:指定META-INF/MANIFEST.MF中的Main-Class
  • <addClasspath>true</addClasspath>:在META-INF/MANIFEST.MF中加上Class-Path并配置依赖包
  • <classpathPrefix>lib/</classpathPrefix>:指定依赖包所在的目录

例如,下面是一个通过maven-jar-plugin插件生成的MANIFEST.MF文件片段:

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

只是生成MANIFEST.MF文件还不够,maven-dependency-plugin插件用于将依赖包拷贝到<outputDirectory>${project.build.directory}/lib</outputDirectory>指定的位置,即lib目录下

配置完成后,通过mvn package指令打包,会在target目录下生成jar包,并将依赖包拷贝到target/lib目录下:

target
  lib
     commons-logging-1.2.jar
     commons-io-2.4.jar
  test.jar

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

详见官方文档:https://maven.apache.org/shared/maven-archiver/examples/classpath.html

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

3、使用maven-assembly-plugin插件打包


maven-assembly-plugin是使用最多的打包方式,可以自定义打包结构,也可以定制依赖项等

例如,大数据项目中往往有很多Shell脚本、SQL脚本、Properties及XML配置等,使用maven-assembly-plugin插件可以让输出的结构清晰并标准化

<build>
	<plugins>
 		<plugin>
			<groupId>org.apache.maven.plugins</groupId>
			<artifactId>maven-assembly-plugin</artifactId>
			<version>2.5.5</version>
			<configuration>
				<archive>
					<manifest>
						<mainClass>com.cc.Main</mainClass>
					</manifest>
				</archive>
				<!-- 使用Maven预配置的描述符 -->
				<descriptorRefs>
					<descriptorRef>jar-with-dependencies</descriptorRef>
				</descriptorRefs>
			</configuration>
		</plugin>
	</plugins>
</build>

打包方式为:mvn package assembly:single

assembly:single是Maven插件的一个目标(goal),用于创建一个包含所有依赖的单一可执行jar文件‌。当执行mvn assembly:single命令时,Maven会使用maven-assembly-plugin来创建一个包含项目所有依赖的jar文件,这个jar文件可以直接运行而不需要额外的依赖库‌

打包结果会在target目录下生成一个xx-jar-with-dependencies.jarjar文件,这个文件不但包含了自己项目中的代码和资源,还包含了所有依赖包的内容。所以可以直接通过java -jar来运行

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

<build>
	<plugins>
		<plugin>
			<groupId>org.apache.maven.plugins</groupId>
			<artifactId>maven-assembly-plugin</artifactId>
			<version>2.5.5</version>
			<configuration>
				<archive>
					<manifest>
						<mainClass>com.cc.Main</mainClass>
					</manifest>
				</archive>
				<!-- 使用Maven预配置的描述符 -->
				<descriptorRefs>
					<descriptorRef>jar-with-dependencies</descriptorRef>
				</descriptorRefs>
			</configuration>
			<executions>
				<execution>
					<id>make-assembly</id>
					<!-- 绑定到package生命周期 -->
					<phase>package</phase>
					<goals>
					    <!-- 只运行一次 -->
						<goal>single</goal>
					</goals>
				</execution>
			</executions>
		</plugin>
	</plugins>
</build>

其中<phase>package</phase><goal>single</goal>即表示在执行mvn package打包时使用assembly:single方式,所以可以直接使用mvn package打包

详见官方文档:https://maven.apache.org/plugins/maven-assembly-plugin/examples/single/filtering-some-distribution-files.html

值得注意的是,如果项目中使用到Spring Framework,用该方式打出来的jar包运行时会报错,使用如下方式可以处理

4、使用maven-shade-plugin插件打包


maven-shade-plugin用来打可执行的jar包,也就是所谓的fat jar

maven-shade-plugin插件提供了两大基本功能:一是将依赖jar包打包到当前jar包,二是对依赖的jar包进行重命名,以用于类的隔离

<build>  
    <plugins>  
        <plugin>  
            <groupId>org.apache.maven.plugins</groupId>  
            <artifactId>maven-shade-plugin</artifactId>  
            <version>2.4.1</version>  
            <executions>  
                <execution>  
                    <phase>package</phase>  
                    <goals>  
                        <goal>shade</goal>  
                    </goals>  
                    <configuration>  
                        <transformers>  
                            <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">  
                                <mainClass>com.cc.Main</mainClass>  
                            </transformer>  
                        </transformers>  
                    </configuration>  
                </execution>  
            </executions>  
        </plugin>  
    </plugins>  
</build>  

配置完成后,执行mvn package即可打包。在target目录下会生成两个jar包,注意不是original-xx.jar文件,而是另外一个。与maven-assembly-plugin相同,生成的jar文件包含了所有依赖,所以可以直接运行

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

<build>  
    <plugins>  
        <plugin>  
            <groupId>org.apache.maven.plugins</groupId>  
            <artifactId>maven-shade-plugin</artifactId>  
            <version>2.4.1</version>  
            <executions>  
                <execution>  
                    <phase>package</phase>  
                    <goals>  
                        <goal>shade</goal>  
                    </goals>  
                    <configuration>  
                        <transformers>  
                            <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">  
                                <mainClass>com.cc.Main</mainClass>  
                            </transformer>  
                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">  
                                <resource>META-INF/spring.handlers</resource>  
                            </transformer>  
                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">  
                                <resource>META-INF/spring.schemas</resource>  
                            </transformer>  
                        </transformers>  
                    </configuration>  
                </execution>  
            </executions>  
        </plugin>  
    </plugins>  
</build>  

详见官方文档:https://maven.apache.org/plugins/maven-shade-plugin/examples/attached-artifact.html

5、Maven自定义打包方式


Maven自定义打包方式示例如下:

<build>
    <finalName>${project.artifactId}</finalName>
    <plugins>
        <!-- 解决资源文件编码问题 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-resources-plugin</artifactId>
            <version>3.0.2</version>
            <configuration>
                <encoding>UTF-8</encoding>
            </configuration>
        </plugin>
        <!-- 设置编译版本 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>3.1</version>
            <configuration>
                <source>1.8</source>
                <target>1.8</target>
                <encoding>UTF-8</encoding>
                <skip>true</skip>
            </configuration>
        </plugin>
        <!-- 打包 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-assembly-plugin</artifactId>
            <version>2.6</version>
            <configuration>
				<archive>
					<manifest>
						<mainClass>com.cc.Main</mainClass>
					</manifest>
				</archive>
				<!-- 使用Maven预配置的描述符 -->
				<descriptorRefs>
					<descriptorRef>jar-with-dependencies</descriptorRef>
				</descriptorRefs>
			</configuration>
            <executions>
                <execution>
                    <id>make-assembly</id>
                    <!-- 绑定到package生命周期 -->
                    <phase>package</phase>
                    <goals>
                        <!-- 只运行一次 -->
                        <goal>single</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

其中,我们也可以配置描述符文件自定义打包内容,详情参考文末参考文章

<configuration>
    <!-- 配置描述符文件 -->
    <descriptors>
        <descriptor>src/main/assembly/assembly.xml</descriptor>
    </descriptors>
</configuration>


参考文章:
https://cloud.tencent.com/developer/article/2089955
https://developer.aliyun.com/article/1378456

Maven是一个常用的项目管理和自动化构建工具,它使用POM(Project Object Model,项目对象模型)描述项目信息和配置,通过插件化的架构提供了诸如项目依赖管理、编译、测试、打包等一系列构建过程的统一管理和自动化执行的功能。 Maven打包方式是指打包项目的输出文件类型和格式,在Maven中,打包方式通过在POM文件中指定packaging元素来确定。常见的打包方式有jar、war、ear、pom等几种。 1. jar打包方式 jar是Java Archive的缩写,是一种基于ZIP格式的归档文件,jar包通常用于打包单个可执行程序或类库,被广泛应用于Java开发中。在Maven中,使用jar打包方式会将项目的源代码编译为class文件,然后打包成一个jar文件并输出到target目录中。 2. war打包方式 war是Web Application Archive的缩写,是一种基于ZIP格式的Web应用程序归档文件,war包包含了Web应用程序所需要的所有资源和配置文件。在Maven中,使用war打包方式会将项目的WEB-INF目录、Web.xml文件和相关类库打包成一个war文件并输出到target目录中。 3. ear打包方式 ear是Enterprise Archive的缩写,是一种基于ZIP格式的企业级应用程序归档文件,ear包含了由多个war、jar和可能的其他组件(如数据库脚本等)组成的企业级应用程序。在Maven中,使用ear打包方式会将多个项目的jar、war等资源打包成一个ear文件并输出到target目录中。 4. pom打包方式 pom打包方式是一种特殊的打包方式,它通常被用于父项目或聚合项目。在Maven中,使用pom打包方式不会生成任何输出文件。 总的来说,Maven打包方式可以根据项目的需要进行选择和配置,能够实现项目快速构建、统一管理和自动化部署等一系列优秀的功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值