maven-mvnd
基本介绍
当我们使用 Maven 构建项目时,通常需要启动 Maven 进行编译、测试和打包等操作,而 Maven 传统的构建太慢了,所以 Maven 新起了一个 maven-mvnd
项目,它的宗旨就是,借鉴来自 Gradle 和 Takari 中的技术以提供更快的 Maven 构建速度。
需要注意的是, maven-mvnd
并不能有独立于 Maven 使用,它只是对 Maven 的一种封装和改进,可以让 Maven 的构建操作更快、更高效。
开源地址如下:
https://github.com/apache/maven-mvnd/
mvnd 底层实现原因:
- mvnd 内嵌了 Maven,安装 mvnd 后无需单独安装 Maven。
- 应用会在一个长驻后台进程中构建,也就是守护进程。
- 一个守护进程实例可以处理 mvnd 客户端的多次连续请求。
- mvnd 客户端是一个使用了 GraalVM 构建的本机可执行文件,与启动传统 JVM 相比,它启动速度更快,占用的内存更少。
- 如果没有空闲的守护进程,它可以并行生成多个守护进程处理构建请求。
mvnd 为什么快的原因:
- 不需要每次构建重新启动 JVM,大大节省时间。
- 持有 Maven 插件类的类加载器缓存在多个构建中,因此插件 jar 只被读取和解析一次。
- 由 JVM 内部的即时 (JIT) 编译器生成的本机代码也被保留。与传统的 Maven 相比,JIT 编译花费的时间更少,在重复构建期间,JIT 优化代码立即可用。
默认情况下,mvnd 使用多个 CPU 内核并行构建模块,使用的核心数由以下公式给出:
Math.max(Runtime.getRuntime().availableProcessors() - 1, 1)
下面是它在 24 核 CPU 机器上的构建图:
构建完成后,控制台会输出完整的 Maven 构建日志。
安装使用
各种系统的最新安装方式见上面的开源地址,那里有很详细的安装手册。
mvnd 和传统的 Maven 使用方式一样,前缀改用 mvnd
即可,后面的选项都是一样的,比如,安装完后,可以使用下面命令验证版本:
$ mvnd --version
可以看到安装的 mvnd 及 Maven 版本信息。
除了 Maven 基本选项, mvnd
还有一些附加选项,比如:
- --status:查看当前所有的守护进程列表;
- --stop:停止所有运行的守护进程;
更多的选项可以使用 mvnd --help
命令查看。
速度测试
栈长分别使用 mvn 及 mvnd 测试下我的 Spring Boot 实战开源学习项目:
https://github.com/javastacks/spring-boot-best-practice
这个开源学习项目有很多个模块,打包还挺费时的,来对比下两者的构建速度:
可以看到实测效果还是挺明显的,同样的机器同样的项目,使用传统的 mvn 需要 1 分多钟,而使用 mvnd 只要 20 多秒就完事了,这对于大工程来说还是挺能提升效率的。
另外,如果你近期准备面试跳槽,建议在Java面试库小程序在线刷题,涵盖 2000+ 道 Java 面试题,几乎覆盖了所有主流技术面试题。
Spring Boot 快速使用 Maven
Maven Wrapper
Spring Boot 也提供了使用 Maven 的相关快捷脚本,即 maven-mvnd
项目中提供的 Maven Wrapper
脚本,它可以免安装、快速使用 Maven、Gradle。
Spring 提供了一站式生成 Spring 应用的网站:
https://start.spring.io/
生成的 Demo 项目如下图所示:
除了常规的项目文件,还有两个特殊的脚本文件:
- mvnw(Linux 版本)
- mvnw.cmd(Windows 版本)
mvnw,全称为:Maven Wrapper,它其实就是来自 Maven 的 maven-mvnd 项目,使用 mvnw 可以快速将 Maven 集成到项目中。
栈长总结一下使用 mvnd 的场景:
- 想省心,不想自己安装配置 Maven 环境;
- 不同的应用,需要使用不同版本的 Maven;比如已有的 Maven 版本不合适,需要使用特定的 Maven 版本;
除了上面所说的 mvnw 脚本,在当前目录下会初始化一个 .mvn/wrapper
目录:
如果 Maven Wrapper 使用的并不是最新的版本,想使用最新的 Maven 3.8.5,可以在 maven-wrapper.properties
配置文件中指定版本:
意味着你如果想用 mvnw,只要复制这些生成的文件到对应的项目目录就行了,然后想用哪个版本,改一改参数即可。
然后在 $USER_HOME/.m2/wrapper
目录中可以看到安装信息:
虽然不用自己另行安装,但本地仓库还是会自动下载对应版本的 Maven,只是省去了自己动手安装的流程。
使用方式
使用 Spring Initializr 网站一键生成的 Spring Boot 项目不需要单独安装 mvnw,生成后默认就带有 mvnw 系列文件,所以就不需要再另行安装 mvnw 了,直接使用即可。
比如,我们切到 demo 项目:
cd demo
运行项目清理安装命令:
./mvnw clean install
就像直接使用 mvn 命令一样,mvnw 只是包了一层而已,底层还是 mvn,所以,如果有多 Maven 版本管理这方面的需求,或者不想自己动手安装 Maven 的场景,这个 mvnw
脚本还是可以用来试试的,但实际工作中很少会使用到。
Gradle 也有类似的包装:
和 Maven 类似,这里就不再撰述了。
IDEA 中设置 mvnw
在 IDEA 中可以设置使用 Maven Wrapper:
如果要设置 Maven 版本信息,需要在当前应用有一个 .mvn 目录及参数定义文件,相关配置文件见 Spring 生成的 demo 项目,或者参考这里:
https://github.com/apache/maven-mvnd/tree/master/.mvn/wrapper
总结
mvnd 和 mvnw 它们两个都是 Maven 的包装工具,两者都是 Maven 的包装,不过 mvnd 是对 Maven 的包装和增强,而 mvnw 仅仅是一个 Maven 包装,两者的功能和作用场景不同。
使用 mvnd 可以比传统的 mvn 有更快的构建速度,它快的关键是它在后台保持一个运行的进程,而不需要每次构建都重新启动,还使用了启动速度更快、内存占用更少的 GraalVM 虚拟机构建。
mvnd 项目还提供了一键安装和运行脚本:mvnw,在 Spring Boot 中也能很轻易的使用,使用 mvnw 可以快速使用 Maven,不同的应用可以使用不同的 Maven 版本进行构建,本地无需手动安装多版本的 Maven。