博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
关于maven的一份小笔记
阅读量:3959 次
发布时间:2019-05-24

本文共 10794 字,大约阅读时间需要 35 分钟。

简介

项目里一直用的 maven,几乎天天和这个“熟知”的工具打交道,但是,最近我发觉自己对 maven 了解的还不够,例如,什么是 goal?什么是 phase?等等。趁着最近有时间,把大致看了一遍,并且做做笔记,也就形成了这篇博客。

本文主要讲解以下内容:

  1. 什么是 maven?maven有什么用?
  2. 安装和使用 maven
  3. maven 的构建生命周期
  4. 配置 maven
  5. 常见问题(持续更新)

什么是maven?maven 有什么用?

这两个问题,很多文章都有说到,但是,大部分都是翻译了官网的这句笼统的话,看了和没看一样。

Apache Maven is a software project management and comprehension tool. Based on the concept of a project object model (POM), Maven can manage a project's build, reporting and documentation from a central piece of information.

以下是我的个人总结,可能稍微好理解一点。

首先,maven 是一个工具,用来帮助我们简化标准化项目的构建,主要分成四点:

  1. 如何描述一个项目。我们可以简单地用一个坐标(groupId、artifactId、version)来描述一个项目。
  2. 将项目的构建分为哪些阶段。maven 将项目的构建过程标准化,划分为多个有序的阶段,例如,默认生命周期大致包括:编译、测试、打包、安装、部署等。
  3. 如何发布和共享项目。maven 项目的发布和共享基于仓库坐标两个基础,我们可将项目发布到仓库,其他人可以通过项目的坐标从仓库中获取这个项目。
  4. 如何处理项目间的关系。我们可以在 pom.xml 配置对应的坐标来依赖其他的项目,而不需要手动地将众多的 jar 包添加到 classpath 中。

下载、安装

项目环境

maven:3.6.3

操作系统:win10

JDK:8u231

下载、安装

进入,根据自己的操作系统和 JDK 选择合适的 maven 版本,这里我们也可以选择下载二进制安装包或者源码包。这里我选择版本 3.6.3 的二进制安装包。

img_maven_download

将下载的 .zip 文件解压,可以看到以下的目录结构:

进行到这一步可以说 maven 已经安装好了,只是我们还需要进行简单的配置。

环境配置

首先,因为 maven 是由 Java 编写,需要 JDK 才能运行,所以,我们必须保证配置好了 JAVA_HOME 的环境变量。这个我就不展开了。

然后,将解压文件中的 bin 目录添加到 Path 的环境变量中(我的电脑(右键属性)->高级系统设置->环境变量),如下所示:

img_maven_environment_variable

测试

在任一位置打开命令行,输入:mvn -vmvn --version,显示以下内容,说明安装完成。

img_maven_mvn_v

简单使用maven

安装完成后,下面通过一个简单的例子来模拟构建 maven 项目的过程。

生成maven项目

cmd 进入到你想要存放项目的文件夹,输入以下命令:

mvn archetype:generate -DgroupId=com.mycompany.app -DartifactId=my-app -DarchetypeArtifactId=maven-archetype-quickstart -DarchetypeVersion=1.4 -DinteractiveMode=false

在这个命令中,archetype:generate为一个goal(后面展开分析),而后面那些都是执行这个goal所需的参数。

如果是你的 maven 是刚安装的,这个命令可以会执行比较久,因为 maven 需要将所需的软件包或其他文件下载到你的本地仓库(默认在 ${user.home}/.m2/repository 目录下)。如果出现连接超时等情况,可以尝试多执行几次(可以将settings.xml的仓库镜像配置为其他地址来提高下载速度)。

执行完这个命令,可以看到指定文件夹下生成了一个 maven 项目,cd my-app,它的文件结构如下:

my-app|-- pom.xml`-- src    |-- main    |   `-- java    |       `-- com    |           `-- mycompany    |               `-- app    |                   `-- App.java    `-- test        `-- java            `-- com                `-- mycompany                    `-- app                        `-- AppTest.java

其中,

src/main/java目录用来放项目的代码,这些代码将会被编译并打包。

src/test/java目录用来放项目的测试代码,这些代码仅进行编译运行,不打包。

如果我们想要添加一些资源文件,可以在src/main目录下创建一个resource目录,这些资源最终也会被打包到项目根目录下。

my-app|-- pom.xml`-- src    |-- main    |   |-- java    |   |   `-- com    |   |       `-- mycompany    |   |           `-- app    |   |               `-- App.java    |   `-- resources    |       `-- META-INF    |           `-- application.properties    `-- test        `-- java            `-- com                `-- mycompany                    `-- app                        `-- AppTest.java

pom.xml则包含了构建这个项目的配置信息,后面我们将重点介绍这个文件。

4.0.0
com.mycompany.app
my-app
1.0-SNAPSHOT
my-app
http://www.example.com
UTF-8
1.7
1.7
junit
junit
4.11
test

上面生成的项目中包含了两个类,其中,App.java 有一个打印 Hello World! 的 main 方法。接下来我们尝试将项目打包并运行。

构建项目

maven 将构建和部署项目的过程定义成了很多个有序的phase(可以理解为步骤),我们可以执行以下命令来打包项目。

mvn package

这个命令将执行package之前的phase,如validatecompiletest等,以及执行package本身。严格上来讲,实际上执行的不是phase,而是绑定在这些phase上的goal,后面会展开讲解。

执行成功后,项目根目录生成了一个 target 文件夹,里面就有打包好的 my-app-1.0-SNAPSHOT.jar。

my-app\target│  my-app-1.0-SNAPSHOT.jar│├─classes│  └─com│      └─mycompany│          └─app│                  App.class│├─generated-sources│  └─annotations├─generated-test-sources│  └─test-annotations├─maven-archiver│      pom.properties│├─maven-status│  └─maven-compiler-plugin│      ├─compile│      │  └─default-compile│      │          createdFiles.lst│      │          inputFiles.lst│      ││      └─testCompile│          └─default-testCompile│                  createdFiles.lst│                  inputFiles.lst│├─surefire-reports│      com.mycompany.app.AppTest.txt│      TEST-com.mycompany.app.AppTest.xml│└─test-classes    └─com        └─mycompany            └─app                    AppTest.class

运行项目

接下来就是运行 jar 包了,在命令行输入

java -cp target/my-app-1.0-SNAPSHOT.jar com.mycompany.app.App

执行完成,可以看到打印:

Hello World!

maven 的构建生命周期

构建生命周期( build lifecycle )是 maven 的核心理论基础之一,它将项目的构建过程标准化。

maven 有三个独立的生命周期:

  1. 默认生命周期。用来定义项目构建的过程。
  2. 清理生命周期。用来定义项目清理的过程。
  3. 站点生命周期。用来定义项目站点发布的过程。

phases

一个生命周期包括了许多具体的phase(可以理解为步骤),如下:

默认生命周期

一般我们接触比较多的是默认生命周期,它主要包括以下过程:

  • validate - 校验项目是一个正确的 maven 项目
  • compile - 编译代码
  • test - 测试src/test/java中的方法,src/test/java的内容仅作为测试使用,不会进行打包或部署
  • package - 将项目打包为可执行的 jar、war 等二进制软件包。
  • install - 将软件包安装到本地仓库
  • deploy - 将软件包部署到远程仓库

除了这几个常用的phase,还有initializegenerate-sourcesprocess-sources等等,需要注意一点,当我们执行某个阶段的命令时,类似 pre-*, post-*, or process-* 的阶段一般只是产生中间结果,并不会对最终构建结果产生影响。

清理生命周期

pre-clean - 在清理项目前执行一些东西

clean- 清理项目,例如删除 target 包

post-clean- 在清理项目后执行一些东西

站点生命周期

pre-site - 在生成站点文档前执行一些东西

site - 生成站点文档

post-site - 在生成站点文档后、部署站点文档前执行一些东西

site-deploy - 部署站点文档

goals

在下面这个 maven 命令中,archetype:generate称之为一个goal,后面那些都是执行这个goal所需的参数。

mvn archetype:generate -DgroupId=com.mycompany.app -DartifactId=my-app -DarchetypeArtifactId=maven-archetype-quickstart -DarchetypeVersion=1.4 -DinteractiveMode=false

至于goal,对应的是某个插件的某个方法,执行这个命令,我们可以在界面中看到执行的是maven-archetype-plugin插件的generate方法

img_maven_goal01

bindings

和上面这个命令不同,下面的这个命令我们并没有传入goal,传入的是phase

mvn clean

执行这个命令,可以看到,这个命令也是执行了插件的方法,换句话来讲,就是执行了goal

img_maven_phase01

这里就涉及到一个很重要的概念:当我们在命令中指定了phase,执行的并不是phase本身,而是绑定在phase上面的goal,绑定的goal数量可以是一个也可以是多个

下面是官方给的部分bindingphasegoal的绑定关系主要和项目的packaging配置有关。

Phase plugin:goal
process-resources resources:resources
compile compiler:compile
process-test-resources resources:testResources
test-compile compiler:testCompile
test surefire:test
package ejb:ejb or ejb3:ejb3 or jar:jar or par:par or rar:rar or war:war
install install:install
deploy deploy:deploy

这些绑定关系,在${MAVEN_HOME}\lib\maven-core-3.6.3\META-INF\plexus\ default-bindings.xml中定义。

mvn [phase]命令的运行

当我们执行phase命令时,在执行指定phase之前,会先有序地执行指定phase之前的phase以及它本身。例如,我执行mvn package,会出现下面的信息:

img_maven_phase02

package之前的phase,包括compiletest等都会被执行,而且是有序的。

settings.xml

pom.xml不同,settings.xml用于全局地配置 maven,而不是配置具体的项目。我们可以在${maven.home}/conf/目录下找到这个文件。

settings.xml文件主要包含以下节点:

这个文件的配置可以参考。这里我补充下serversmirrorsprofiles这三个节点的内容。

servers–配置仓库认证授权信息

servers用于配置仓库(包括下载项目和部署项目的仓库)的认证授权信息,例如,用户密码等。

在具体项目中,我们可以在pom.xml中的repositoriespluginRepositoriesdistributionManagement节点配置用于下载项目和部署项目的仓库,但是我们不能把认证授权的信息放在pom.xml文件中,于是servers就发挥了作用。

...
server001
my_login
my_password
${user.home}/.ssh/id_dsa
some_passphrase
664
775
...

Mirrors–配置仓库的镜像

Mirrors用于配置下载项目的仓库镜像。前面说过,国内使用 maven 的中央仓库下载项目比较慢,甚至会出现超时失败的情况,这时,我们就可以通过配置镜像来提高传输速度。在此之前,我们需要区分镜像和仓库两个概念,以下这篇文章作出了很好的解释。

在下面这个例子中,我们使用阿里云的镜像来请求 maven 的中央仓库,注意,mirrormirrorOf节点必须指定仓库的 id,当然,这里还支持多种形式。例如,<mirrorOf>*</mirrorOf>表示匹配所有远程仓库;<mirrorOf>repo1,repo2</mirrorOf>表示匹配仓库 repo1 和 repo2,使用逗号分隔多个远程仓库;<mirrorOf>*,!repo1</miiroOf> 匹配所有远程仓库,repo1 除外,使用感叹号将仓库从匹配中排除。

...
alimaven
aliyun maven
http://maven.aliyun.com/nexus/content/groups/public/
central
...
central
Central Repository
https://repo.maven.apache.org/maven2
default
false
...

补充下,mirror节点对repositoriespluginRepositoriesdistributionManagement均生效。

profiles

profiles:提供了一组可选的配置,我们可以根据不同的环境选择激活哪一套配置,它包括:activationrepositoriespluginRepositoriesproperties四个节点。其中,activation节点用于配置该profile在什么环境下才能激活。

...
test
false
1.5
Windows XP
Windows
x86
5.1.2600
mavenVersion
2.0.3
${basedir}/file2.properties
${basedir}/file1.properties
${user.home}/our-project
codehausSnapshots
Codehaus Snapshots
false
always
warn
true
never
fail
http://snapshots.maven.codehaus.org/maven2
default
...
...
...

注意,如果settings.xml的某个profile被激活,那么,它的配置将覆盖pom.xml中相同 id 的仓库以及相同名称的 property

pom.xml

pom.xml几乎包含了对 maven 项目的所有描述信息,包括项目的坐标、依赖关系、构建配置等等。这个文件非常重要,官网有这么一句话,在 maven 的世界里,一个完整的项目可以不包含任何的代码,而只需要一个pom.xml

pom.xml的节点如下:

4.0.0
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...
...

关于pom.xml文件的内容,就不详细展开了,可参考。这里介绍下两个比较重要的概念。

scope

scopedependency的子节点,用于设置以下两个内容:

  1. 依赖是否在(测试)编译、(测试)运行等时机加入 classpath。
  2. 限制依赖的传递性。(假设当前项目为 A,它依赖了 B,如果 C 依赖了 A,则 C 也会依赖 B。可以看出,A 将自己对 B 的依赖传递给了 C)

maven 提供了五种scope给我们选择,如下。

scope 编译期 运行期 测试编译期 测试运行期 依赖传递
compile
provided × ×
runtime × ×
test × × ×
system

通过dependency的子节点optional可以改变传递性。system对应的依赖不会从仓库获取,而是从systemPath指定的路径中获取

super pom

pom 文件可以通过<parent>节点来继承其他项目的配置信息,而且,和 Java 的对象默认继承 Object 一样,pom 文件默认会去继承 super pom,该 pom 文件的内容见:

常见问题

中央仓库没有的依赖,怎么获取

当我们的项目需要依赖某个在中央仓库中不存在的依赖,例如,oracle的驱动包,我们可以采用三种解决方案:

  1. 将依赖的项目安装到本地仓库。命令如下:
mvn install:install-file -Dfile=non-maven-proj.jar -DgroupId=some.group -DartifactId=non-maven-proj -Dversion=1 -Dpackaging=jar
  1. 将依赖的项目安装到私服。命令如下:
deploy:deploy-file -Dfile=non-maven-proj.jar -DgroupId=some.group -DartifactId=non-maven-proj -Dversion=1 -Dpackaging=jar
  1. 使用system作用域指定包的路径。
some.group
non-maven-proj
1.0
system
${java.home}/lib/non-maven-proj.jar

参考资料

相关源码请移步:

本文为原创文章,转载请附上原文出处链接:

你可能感兴趣的文章
杭电ACM——2069,Coin Change(DP)
查看>>
杭电ACM——2074,叠筐
查看>>
北大ACM——3616,Milking Time(DP)
查看>>
杭电ACM——2076,夹角有多大
查看>>
牛客练习赛43——B Tachibana Kanade Loves Probability(暴力,思维)
查看>>
牛客第十七届上海大学程序设计春季联赛——E CSL 的魔法(贪心)
查看>>
杭电ACM——1028,Ignatius and the Princess III(母函数)
查看>>
杭电ACM——1171,Big Event in HDU(母函数)
查看>>
杭电ACM——6491,时间间隔(思维)
查看>>
杭电AC——1085,Holding Bin-Laden Captive!(母函数)
查看>>
杭电ACM——2110,Crisis of HDU(母函数)
查看>>
杭电AM——2152,Fruit(母函数)
查看>>
杭电ACM——2566,统计硬币(DP)
查看>>
堆栈(数据结构)
查看>>
队列(数据结构)
查看>>
杭电ACM——1251,统计难题(Trie树)
查看>>
牛客网哈尔滨工程大学第十四届程序设计竞赛(同步赛)—— 小蚂蚁过马路(思维)
查看>>
牛客网哈尔滨工程大学第十四届程序设计竞赛(同步赛)—— 苹果手链(水题)
查看>>
杭电ACM——6518,Clumsy Keke(暴力+思维)
查看>>
杭电ACM——6512,Triangle(暴力 / 思维)
查看>>