maven生命周期指的是特定的一组有序的阶段。软件构建过程经过多年来的总结和发展,其实总归会有几个阶段,比如编译,测试,打包等,而且内置是有一定的顺序。所以maven所做的就是把这些经验性的总结吸收到自己的理念中,规范出一套通用的流程,这样maven的使用者就不用再重复地关心这个过程。

maven最终总结出3个生命周期,每一个对应不同的使用场景,它们是default,clean,site。生命周期是由一个一个阶段(phase)组成,比如clean生命周期包括了pre-clean,clean和post-clean。每一个阶段相当于是生命周期中的子任务。阶段是由严格的顺序关系的,我们可以选择执行哪一个阶段,而且,如果想要执行后面的阶段,前面的阶段也会执行。这是合理的,因为我们如果想要执行test阶段,那么代码肯定需要先compile啊,所以test之前的compile阶段也会执行。

另一个概念就是插件和目标了。maven本身是一个十分轻量级的产品,据说只有3M。那么maven如何完成繁重的任务呢?答案就是插件,插件也有坐标,但是类型是maven-plugin。之前所说的生命周期和阶段都是一个逻辑概念,maven运行的时候并不是直接对应这些概念的。实际上,每一个阶段最终都是交给相应的插件来完成的。这个叫做插件绑定。maven默认地为每一个阶段绑定了插件。但是插件只是一个工具或者器具,我们可以使用一个插件完成好多任务,每一个插件的任务叫做目标。所以,要运行一个插件必须指定目标才行,maven的每一个阶段实际上对应的是<插件:目标>。

我们可以使用help插件来查看任何插件,比如compile:

mvn help:describe -Dplugin=org.apache.maven.plugins:maven-compiler-plugin

通过-D来指定插件的坐标,这里看一哈compile插件:

【maven】生命周期和插件_maven

其实上面就是调用了help插件的describe的目标。

所以,生命周期和阶段是maven的逻辑概念,插件和目标是maven的执行的单元,他们是按照生命周期来组织的。

我们运行maven可以使用两种方式:

1.指定阶段,比如mvn clean,这个就是指定了clean生命周期的clean阶段。或者mvn test,是default生命周期的test阶段。

2.指定插件目标,比如mvn compiler:compile,就是compiler插件的compile目标,和mvn compile一样。

下面看下maven默认的插件绑定:

【maven】生命周期和插件_插件_02

所以,mvn clean虽然会执行pre-clean和clean两个阶段,但是由于pre-clean阶段没有任何插件绑定,实际上只执行了clean阶段的任务。

【maven】生命周期和插件_maven_03

这是default的,只列出了核心的部分。

那么我们如何设定自己的插件绑定?

比如默认只会把class文件打包,现在想把source文件也打包。

maven-source-plugin的jar-no-fork目标就可以完成这个任务。

使用mvn archetype:generate创建一个quick-start的项目。

然后再pom的build标签下配置这个插件。pom如下:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<groupId>com.liyao</groupId>
<artifactId>cvi</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>jar</packaging>

<name>cvi</name>
<url>http://maven.apache.org</url>

<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>

<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-source-plugin</artifactId>
<version>3.0.1</version>
<executions>
<execution>
<id>123</id>
<phase>test</phase>
<goals>
<goal>
jar-no-fork
</goal>
</goals>
</execution>
</executions>
</plugin>

</plugins>

</build>
</project>

我们把插件绑定到了test阶段,虽然不是一个好主意,但是为了演示。

然后运行,mvn test:

【maven】生命周期和插件_插件_04

可以看到生成了一个sources的jar文件:

【maven】生命周期和插件_插件_05

说明该插件配置生效了。

这个例子参考自《maven实战》,很好的书籍。