目录
- 写作原由
- Maven的安装(略)
- POM文件
- POM文件的分类
- POM文件的继承和重写
- POM文件的组件坐标(标识)
- 几个简单的例子
- 举例一、继承不如组合
- 举例二、可选依赖
- 举例三、排除依赖
- 术语说明
写作原由
说起Maven,它不是一个新的东西,大家多少都了解一些。之前为了了解的更多一些,我看了一本中文的Maven介绍书籍,但是那时一方面是经验尚浅,对一些概念的理解不深,一方面是书写的思路不是很清晰,后续没有相关总结,现在说不上一二。近期对Maven重新找了一本比较好的外文书籍,初读感觉很清爽,打算做一下读书笔记。(这里不是崇洋媚外,^_^)所以这里的原创其实是抱着了巨人的大腿。计划写八、九篇,大约一二周出一篇。
第一讲,围绕Maven的核心要素POM(Project Object Model)。这一讲是纲领,也许有些介绍看起来是粗粒度的,但是浓缩的都是精华,抽取提炼,更容易看出Maven工程的骨干。
Maven的安装(略)
这个部分本应该包含Maven的下载安装、本地仓库的配置、远程代理的设置、IDE的配置,但是相关资料很容易找到,这里不想多说了,百度一下,很容易的。
POM文件
pom.xml是Maven工程的描述文件,就像一般的JavaEE web工程总有一个web.xml。 一般的pom文件包含以下内容:
<project>
<!-- 本项目切身描述信息 -->
<parent>...</parent>
<modelVersion>4.0.0</modelVersion>
<groupId>...</groupId>
<artifactId>...</artifactId>
<version>...</version>
<packaging>...</packaging>
<name>...</name>
<description>...</description>
<url>...</url>
<inceptionYear>...</inceptionYear>
<licenses>...</licenses>
<organization>...</organization>
<developers>...</developers>
<contributors>...</contributors>
<!-- 依赖信息 -->
<dependencies>...</dependencies>
<dependencyManagement>...</dependencyManagement>
<modules>...</modules>
<properties>...</properties>
<!-- 构建信息 -->
<build>...</build>
<reporting>...</reporting>
<issueManagement>...</issueManagement>
<ciManagement>...</ciManagement>
<mailingLists>...</mailingLists>
<scm>...</scm>
<prerequisites>...</prerequisites>
<!-- 仓库配置信息 -->
<repositories>...</repositories>
<pluginRepositories>...</pluginRepositories>
<distributionManagement>...</distributionManagement>
<profiles>...</profiles>
</project>
这里只给出了一级的标签,这些就是龙骨。对于其中的标签后续会展开。这些标签最好是记住,根据代码中的分块,很容易记忆的。
POM文件的分类
- 超级pom
- 父pom
- 子pom
继承是面向对象语言的一大核心,妥善的处理可以减少代码的冗余和重复。Maven底层是用Java写的,自然不能缺少这一特性。子pom的配置内容可以继承父pom文件的相关配置。子pom文件的parent标签指明父pom文件的位置。parent的具体内容如下:
<parent>
<groupId>org.wso2.carbon</groupId>
<artifactId>platform-parent</artifactId>
<version>4.2.0</version>
<relativePath>../parent/pom.xml</relativePath>
</parent>
其中relativePath是父pom.xml相对于本工程模块的相对路径。这个默认值是…/pom.xml。举例来说,如果是下面展示的路径,模块B使用A的作为父pom,就可以省略relativePath配置。
├── Module_A
│├── Module_B
│└── pom.xml
└── pom.xml
当parent标签省略时,项目默认继承超级pom文件。这个就像是Java的默认继承Object类。超级pom文件在MAVEN_HOME/lib/maven-model-builder-3.2.3.jar - org/apache/maven/model/pom-4.0.0.xml。现在你应该明白一级标签modelVersion的含义了。
POM文件的继承和重写
使用以下命令可以查看当前pom文件继承父文件后的所有属性。
$ mvn help:effective-pom
如果需要重写(Overide)父文件的某一项配置,可以在子文件中个性化对应的配置。但是注意相应的标签级次要完整,必要的标签不可少,如下面的artifactId,这个就是下面说的Maven坐标。
<build>
<pluginManagement>
<plugins>
<plugin>
<artifactId>maven-release-plugin</artifactId>
<version>2.5</version>
</plugin>
</plugins>
</pluginManagement>
</build>
POM文件的组件坐标(标识)
<groupId>com.packt</groupId>
<artifactId>sample-one</artifactId>
<version>1.0.0</version>
上面的配置,多数人知道分别代表公司或组织的域名、工程名称、版本号。但是你知道Maven是怎么定位对应的jar文件(或war)的吗?Maven会在本地仓库的USER_HOME/.m2/repository/com/packt/sample-one/1.0.0/ 下面寻找对应的文件。
当然,对于插件的定位,不需要groupId,具体的原因后续插件的专讲中会提到。
几个简单的例子
这部分用几个简单的例子,来说明pom文件继承的妙用。
举例一、继承不如组合
Java单继承的缘故,有时候继承不如组合更灵活。
对于大型项目的依赖版本的管理,一种做法是通过父pom文件的dependencyManagement统一配置。即如下图所示,父pom的版本属性由配置约定,子pom继承时,无需配置版本,但是依然可以重写父类的版本。
<!-- 父pom -->
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-transport-mail</artifactId>
<version>${axis2-transports.version}</version>
</dependency>
<dependency>
<groupId>org.apache.ws.commons.axiom.wso2</groupId>
<artifactId>axiom</artifactId>
<version>${axiom.wso2.version}</version>
</dependency>
</dependencies>
</dependencyManagement>
<properties>
<axis2.wso2.version>1.6.1.wso2v10</axis2.wso2.version>
</properties>
<!-- 子pom -->
<dependencies>
<dependency>
<groupId>org.apache.axis2.wso2</groupId>
<artifactId>axis2</artifactId>
</dependency>
<dependency>
<groupId>org.apache.ws.commons.axiom.wso2</groupId>
<artifactId>axiom</artifactId>
</dependency>
</dependencies>
但组合的处理是这样的,把共有的几个依赖抽取出来,单独作为一个pom文件,注意下面的packaging属性是pom。用到依赖的pom文件就能简化:
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>axis2-client</artifactId>
<version>1.0.0</version>
<packaging>pom</packaging>
<dependencies>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-kernel</artifactId>
<version>1.6.2</version>
</dependency>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-adb</artifactId>
<version>1.6.2</version>
</dependency>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-transport-http</artifactId>
<version>1.6.2</version>
</dependency>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-transport-local</artifactId>
<version>1.6.2</version>
</dependency>
<dependency>
<groupId>org.apache.axis2</groupId>
<artifactId>axis2-xmlbeans</artifactId>
<version>1.6.2</version>
</dependency>
</dependencies>
</project>
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>my-axis2-client</artifactId>
<version>1.0.0</version>
<dependencies>
<dependency>
<groupId>com.packt</groupId>
<artifactId>axis2-client</artifactId>
<version>1.0.0</version>
</dependency>
</dependencies>
</project>
举例二、可选依赖
假设这样的一个场景,我们需要基于两种OSGi运行时进行编程,因而我们对两种的各自APIs都做了调用。但是实际程序运行的时候,依据实际底层的OSGi,只有一类的APIs在被调用。简言之,运行时(runtime)有一个jar,但是编译的时候却需要两个jar。对此,加入optional可选依赖。
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>osgi.client</artifactId>
<version>1.0.0</version>
<dependencies>
<dependency>
<groupId>org.eclipse.equinox</groupId>
<artifactId>osgi</artifactId>
<version>3.1.1</version>
<scope>compile</scope>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.apache.phoenix</groupId>
<artifactId>phoenix-core</artifactId>
<version>3.0.0-incubating</version>
<scope>compile</scope>
<optional>true</optional>
</dependency>
</dependencies>
</project>
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>my.osgi.client</artifactId>
<version>1.0.0</version>
<dependencies>
<!-- 对实际使用的明确指定 -->
<dependency>
<groupId>org.eclipse.equinox</groupId>
<artifactId>osgi</artifactId>
<version>3.1.1</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>com.packt</groupId>
<artifactId>osgi.client</artifactId>
<version>1.0.0</version>
<scope>compile</scope>
</dependency>
</dependencies>
</project>
举例三、排除依赖
有两个项目分别是这样的:
<!-- 项目1 -->
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>jose</artifactId>
<version>1.0.0</version>
<dependencies>
<dependency>
<groupId>com.nimbusds</groupId>
<artifactId>nimbus-jose-jwt</artifactId>
<version>2.26</version>
</dependency>
<dependency>
<groupId>net.minidev</groupId>
<artifactId>json-smart</artifactId>
<version>1.0.9</version>
</dependency>
</dependencies>
</project>
<!-- 项目2 -->
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>jose.ext</artifactId>
<version>1.0.0</version>
<dependencies>
<dependency>
<groupId>com.nimbusds</groupId>
<artifactId>nimbus-jose-jwt</artifactId>
<version>2.26</version>
</dependency>
<dependency>
<groupId>net.minidev</groupId>
<artifactId>json-smart</artifactId>
<version>1.1.1</version>
</dependency>
</dependencies>
</project>
虽然他们的依赖json-smart版本不同,但是在各自的项目里,他们各自为政,不会出现问题。但是如果有第三个项目同时引用以上的项目:
<!-- 项目3 -->
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>jose.war</artifactId>
<version>1.0.0</version>
<version>war</version>
<dependencies>
<!-- 项目1 -->
<dependency>
<groupId>com.packt</groupId>
<artifactId>jose</artifactId>
<version>1.0.0</version>
</dependency>
<!-- 项目2 -->
<dependency>
<groupId>com.packt</groupId>
<artifactId>jose.ext</artifactId>
<version>1.0.0</version>
</dependency>
</dependencies>
</project>
由于Maven的传递性,各依赖项目原始所需的jar包也会引入。最终会引入json-smart 1.1.1的jar。但是如果我们想使用1.0.9的jar,我们使用exclusions来配置排除掉传递进来的属性。
<project>
<modelVersion>4.0.0</modelVersion>
<groupId>com.packt</groupId>
<artifactId>jose.war</artifactId>
<version>1.0.0</version>
<version>war</version>
<dependencies>
<dependency>
<groupId>com.packt</groupId>
<artifactId>jose</artifactId>
<version>1.0.0</version>
</dependency>
<dependency>
<groupId>com.packt</groupId>
<artifactId>jose.ext</artifactId>
<version>1.0.0</version>
<exclusions>
<exclusion>
<groupId>net.minidev</groupId>
<artifactId>json-smart</artifactId>
</exclusion>
</exclusions>
</dependency>
</dependencies>
</project>
术语说明
对于artifact的翻译,本讲是“项目”,但是这个其实就是一个区分用的字符标识。准确的说是jar包(或war包等)的名称(groupId组成路径,artifactId和version组成jar文件名称)。当然,你翻译为工艺品也没毛病,愿每个项目都是我们开发的工艺品。