Maven是什么?
Maven,这个单词来源于犹太语,意味着知识的积累。最初在JakartaTurbine项目中用来简化构造项目流程。最终,形成基于Java项目的构建和管理的工具。
安装与配置
安装比较简单,官网下载,配置环境即可。这里推荐一下,安装之后配置仓库镜像,可以加快访问速度。
编辑settings.xml,在</mirrors>之前添加
使用其他工具创建项目也如出一辙。
Maven项目结构
Maven基本概念
POM(ProjectObjectModel)
POM的全称是ProjectObjectModel,用通俗点的话说就是对要构建的项目进行建模,将要构建的项目看成是一个对象(Object)。
我们可以用我们熟悉的一个Java代码来描述一下这个对象。
Lifecycle(生命周期)
在Maven中一次构建过程就是一个Lifecycle,这个Lifecycle分为多个阶段,每个阶段叫做Phase。有三种构建的生命周期,分别是default、clean、site。default用于部署项目,clean用于清理项目,而site用于创建项目文档。
defaultLifecycle包含如下Phase:
- validate:校验项目和一些必要信息是可用的
- compile:编译项目源码
- test:执行源码的单元测试模块
- package:将项目打包,比如打包成jar文件
- verify:对集成测试的结果进行检查,以确保满足质量标准
- install:将软件包安装到本地仓库中
- deploy:将软件包部署到远程仓库
maven常用命令介绍
清理项目产生的临时文件,一般是模块下的target目录
mvnclean
项目打包工具,会在模块下的target目录生成jar或者war等文件
mvnpackage
测试命令或者执行src/test/java下面的测试用例
mvntest
模块安装命令,将打包的jar或者war文件复制到本地仓库,使用-Dmaven.test.skip=true跳过测试
mvninstall
发布命令,将打包的文件发布到maven仓库
mvndeploy
maven多个命令同时使用
mvncleanpackage-U-Dmaven.test.skip=true-Ptest
Phase和Goals
Lifecycle只规定了项目构建的流程,即先执行validate,再compile等一系列Phase,但并没有定义每一个Phase具体做什么。这里Phase的作用类似于Java中的接口,而Phase的具体实现在Goals里面。
一个Phase必须绑定一个或多个Goals,才能执行具体的构建流程。为了让用户不用任何配置就能使用Maven项目,Maven默认为一些核心生命周期的Phase绑定了Goals。
如果需要自定义绑定,可以在pom.xml文件中配置。
Maven核心概念
Maven坐标
Maven坐标主要是为了标识项目的唯一性。由下面几个属性组成:
- groupId:组织或者组织的项目名称
- artifactId:项目中的具体模块
- version:项目版本
- packaging:项目打包方式
- classifier:用于区分从同一POM构建的具有不同内容的构件
Maven仓库
存储管理构件(JAR、WAR等)的地方。
一般分为以下三类:
- 本地仓库:默认是~/.m2/repository目录,可在配置文件中配置其他目录
- 私服:内网的Maven仓库
- 中央仓库:Maven社区提供的仓库,包含大量的常用库
Maven依赖
在pom.xml配置项目依赖,比如
<dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope></dependency>
scope用来控制依赖和编译、测试、运行的classpath的关系。有下面三种关系:
- compile:默认编译依赖范围。对于编译、测试、运行三种classpath都有效
- test:测试依赖范围。只对测试classpath有效
- provided:已提供依赖范围。编译、测试classpath有效,对于运行无效。
- runtime:运行时提供
- test:仅用于测试,且不可传递
- system:类似于provided,但必须显示提供JAR
- import:仅在dependencyManagement下支持。
传递依赖
Maven是通过传递依赖解析JAR包依赖,比如我项目中引入junit,在解析我的项目的时候,不仅仅仅有junit,还有junit依赖的JAR包。
包冲突时怎么产生的?
假设A->B->C->D1,E->F->D2,D1、D2分别为D的不同版本。
如果pom.xml文件中引入了A和E之后,按照Maven传递依赖原则,工程内需要引入的实际JAR包将会有:ABCD1和EFD2,因此D1、D2将会产生包冲突。
如何解决包冲突?
Maven解析pom.xml的时候,同一个JAR包只会保留一个。
对于包冲突,Maven处理策略:
- 最短路径优先:Maven面对D1D2,会选择最短路径的那个,即D2。因为,E->F->D2比A->B->C->D1路径短。
- 最先声明优先:如果路径一样,就选择最先声明的JAR包。
如何移除依赖?
Maven继承
继承为了消除重复。可以把很多相同的配置提取出来。
子模块通过parent标签配置,继承父模块属性。父模块通过dependencyManagement标签进行管理。
多模块版本号可以通过maven命令操作,在项目根目录执行
设置新版本号
mvnversions:set-DnewVersion=0.0.2-SNAPSHOT
回滚设置新版本号操作
mvnversions:revert
提交设置新版本号操作
mvnversions:commit
Maven多模块项目结构,以及Maven属性继承配置等