rodert教你学Maven-实战这一篇就够了(下)

简介: rodert教你学Maven-实战这一篇就够了

rodert教你学Maven-实战这一篇就够了(上):https://developer.aliyun.com/article/1536216

6.什么是仓库?

了解内容,可跳过

Maven在某个统一的位置存储所有项目的共享的构件,这个统一的位置,就称之为仓库。(仓库就是存放依赖和插件的地方)Maven的仓库有两大类:


本地仓库

远程仓库,在远程仓库中又分成了3种:中央仓库、私服、其它公共库。


本地仓库:就是Maven在本机存储构件的地方。maven的本地仓库,在安装maven后并不会创建,它是在第一次执行maven命令的时候才被创建。maven本地仓库的默认位置:在用户的目录下都只有一个.m2/repository/的仓库目录;可以修改。


中央仓库:包含了绝大多数流行的开源Java构件,以及源码、作者信息、SCM、信息、许可证信息等。开源的Java项目依赖的构件都可以在这里下载到。

中央仓库的地址:http://repo1.maven.org/maven2/


私服:是一种特殊的远程仓库,它是架设在局域网内的仓库。

没有使用私服的仓库构件下载

使用私服的仓库构件下载

7.什么是maven坐标?

在平面几何中坐标(x,y)可以标识平面中唯一的一点。

Maven坐标主要组成

groupId:定义当前Maven项目隶属项目、组织

artifactId:定义实际项目中的一个模块

version:定义当前项目的当前版本

packaging:定义该项目的打包方式(pom/jar/war,默认为jar)

groupId、artifactId、version简称为GAV。

Maven为什么使用坐标?


Maven世界拥有大量构件,需要找一个用来唯一标识一个构建的统一规范

拥有了统一规范,就可以把查找工作交给机器


如何获取Maven坐标


推荐一个Maven坐标查询网站:http://mvnrepository.com/

网站上可以搜索具体的组织或项目关键字,之后复制对应的坐标到pom.xml中。如:

8.maven依赖管理

8.1.依赖范围

参考:http://www.yiidian.com/maven/maven-dependence.html

依赖范围scope 用来控制依赖和编译,测试,运行的classpath的关系。具体的依赖范围有如下6种:


compile: 默认编译依赖范围。对于编译,测试,运行三种classpath都有效

test:测试依赖范围。只对于测试classpath有效

provided:已提供依赖范围。对于编译,测试的classpath都有效,但对于运行无效。因为由容器已经提供,例如servlet-api

runtime:运行时提供。例如:jdbc驱动

system:系统范围,自定义构件,指定systemPath;跟provided 相似,但是在系统中要以外部JAR包的形式提供,maven不会在repository查找它。

import:只使用在中,表示从其它的pom中导入dependency的配置。

下面是为了解决冲突依赖内容,如果你在写一个Demo,可作为了解部份。

8.2.传递性依赖

假设 C 依赖 B , B 依赖 A ,那么称 C 对 B 是第一直接依赖, B 对 A 是第二直接依赖, C 对 A 是传递依赖。

对于传递性依赖,依赖的范围如下表:

8.3.可选依赖

在依赖节点dependency中的可以控制当前的依赖是否向下传递;默认值为false,表示向下传递。

【示例】A项目依赖于log4j,然后B项目依赖于A项目;那么如果在A中对log4j依赖的optional配置成false时,B项目中自动传递依赖于log4j。否则反之。

1)项目A配置slf4j的依赖并设置optional为true

 <dependency>
  <groupId>org.slf4j</groupId>
  <artifactId>slf4j-log4j12</artifactId>
  <version>1.6.4</version>
  <!-- 配置为true时不向下传递此依赖,默认为false -->
  <optional>true</optional>
 </dependency>

2)配置项目B依赖于项目A,检查项目B的依赖包

<!-- 依赖于A -->
<dependency>
  <groupId>com.JavaPub</groupId>
  <artifactId>A</artifactId>
  <version>0.0.1-SNAPSHOT</version>
</dependency>

这时发现B项目没有依赖slf4j-log4j12

8.4.依赖冲突

依赖冲突是很常见的问题

  • 如果直接与间接依赖中包含有同一个坐标不同版本的资源依赖,以直接依赖的版本为准(就近原则)

最终A依赖的X的版本为2.0

【比如】:项目A中,依赖了slf4j1.6.4版本的包,通过slf4j1.6.4间接依赖log4j1.2.16版本;如果项目A中直接配置了log4j 1.2.17版本,那么最终的版本为1.2.17。

<dependency>
  <groupId>org.slf4j</groupId>
  <artifactId>slf4j-log4j12</artifactId>
  <version>1.6.4</version>
  <!-- 配置为true时不向下传递此依赖,默认为false -->
  <optional>true</optional>
</dependency>
<dependency>
  <groupId>log4j</groupId>
  <artifactId>log4j</artifactId>
  <version>1.2.17</version>
</dependency>

如果直接依赖中包含有同一个坐标不同版本的资源依赖,以配置顺序下方的版本为准

20200611114003803.jpg


如果间接依赖中包含有同一个坐标不同版本的资源依赖,以配置顺序上方的版本为准


如下对应spring-core的间接依赖的版本号,以struts2-spring-plugin为准

<dependency>
  <groupId>org.apache.struts</groupId>
  <artifactId>struts2-spring-plugin</artifactId>
  <version>2.3.24.1</version>
</dependency>
<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-beans</artifactId>
  <version>4.1.0.RELEASE</version>
</dependency>

8.4.排除依赖

这个在项目中使用频率较高

在pom中的依赖节点中,如果引入的依赖包含了很多其它的传递依赖,而且项目需要的这些依赖的版本和传递依赖的不相符;那么可以在依赖节点中设置排除依赖节点: 然后再添加 ,其里面的内容包括:

①所包含坐标

②排除依赖包中所包含的依赖关系

【注意】不需要添加版本,直接按照类别排除

<dependency>
  <groupId>org.apache.struts</groupId>
  <artifactId>struts2-spring-plugin</artifactId>
  <version>2.3.24.1</version>
  <exclusions>
    <!-- 排除spring-core的传递依赖 -->
    <exclusion>
      <groupId>org.springframework</groupId>
      <artifactId>spring-core</artifactId>
    <eclusion>
  <exclusions>
</dependency>

<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-beans</artifactId>
  <version>4.1.0.RELEASE</version>
</dependency>

9.maven生命周期

9.1.Maven生命周期

Maven生命周期就是为了对所有的构建过程进行抽象和统一;包括项目清理,初始化,编译,打包,测试,部署等几乎所有构建步骤。

Maven有三套相互独立的生命周期,请注意这里说的是“三套”,而且“相互独立”,这三套生命周期分别是:


Clean Lifecycle 在进行真正的构建之前进行一些清理工作。

Default Lifecycle 构建的核心部分,编译,测试,打包,部署等等。

Site Lifecycle 生成项目报告,站点,发布站点。

再次强调它们是相互独立的,可以仅仅调用clean来清理工作目录,仅仅调用site来生成站点。不过也可以直接运行 mvn clean install site 运行所有这三套生命周期。

9.2.clean生命周期

clean生命周期每套生命周期都由一组阶段(Phase)组成,我们平时在命令行输入的命令总会对应于一个特定的阶段。比如,运行mvn clean ,这个的clean是clean生命周期的一个阶段。有clean生命周期,也有clean阶段。cleaan生命周期一共包含了三个阶段:

  • pre-clean 执行一些需要在clean之前完成的工作
  • clean 移除所有上一次构建生成的文件
  • post-clean 执行一些需要在clean之后立刻完成的工作

mvn clean 中的clean就是上面的clean,在一个生命周期中,运行某个阶段的时候,它之前的所有阶段都会被运行,也就是说,mvn clean 等同于 mvn pre-clean clean ,如果我们运行 mvn post-clean ,那么 pre-clean,clean 都会被运行。这是Maven很重要的一个规则,可以大大简化命令行的输入。

9.3.default生命周期

default生命周期default生命周期是Maven生命周期中最重要的一个,绝大部分工作都发生在这个生命周期中。比较重要和常用的阶段如下:

validate

generate-sources

process-sources

generate-resources

process-resources 复制并处理资源文件,至目标目录,准备打包。

compile 编译项目的源代码。

process-classes

generate-test-sources

process-test-sources

generate-test-resources

process-test-resources 复制并处理资源文件,至目标测试目录。

test-compile 编译测试源代码。

process-test-classes

test 使用合适的单元测试框架运行测试。这些测试代码不会被打包或部署。

prepare-package

package 接受编译好的代码,打包成可发布的格式,如 JAR 。

pre-integration-test

integration-test

post-integration-test

verify 运行任何检查,验证包是否有效且达到质量标准。

install 将包安装至本地仓库,以让其它项目依赖。

deploy 将最终的包复制到远程的仓库,以让其它开发人员与项目共享。

运行任何一个阶段的时候,它前面的所有阶段都会被运行,这也就是为什么运行mvn install 的时候,代码会被编译,测试,打包。此外,Maven的插件机制是完全依赖Maven的生命周期的。


9.4.site生命周期

site生命周期包含如下4个阶段:


pre-site 执行一些需要在生成站点文档之前完成的工作

site 生成项目的站点文档

post-site 执行一些需要在生成站点文档之后完成的工作,并且为部署做准备

site-deploy 将生成的站点文档部署到特定的服务器上

这里经常用到的是site阶段和site-deploy阶段,用以生成和发布Maven站点,这是Maven相当强大的功能,Manager比较喜欢,文档及统计数据自动生成,很好看。


10.maven插件使用

参考:http://www.yiidian.com/maven/maven-plugins.html


11.maven继承与聚合

11.1.maven继承

继承为了消除重复,可以把pom中很多相同的配置提取出来;如:grouptId,version等。在使用的时候子工程直接继承父工程的依赖版本号,子工程中不再需要指定具体版本号,方便统一管控项目的依赖版本问题。


父工程设置依赖

<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>cn.sm1234</groupId>
  <artifactId>parent</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  <packaging>pom</packaging>

  <!-- 集中定义依赖版本号 -->
  <properties>
    <junit.version>4.10</junit.version>
    <spring.version>4.1.0.RELEASE</spring.version>
    <slf4j.version>1.6.4</slf4j.version>
  </properties>

  <!-- 版本锁定,当子工程中有需要并且自行添加了具体依赖后才有效 -->
  <dependencyManagement>
    <dependencies>
      <!-- 单元测试 -->
      <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>${junit.version}</version>
        <scope>test</scope>
      </dependency>

      <!-- Spring -->
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>${spring.version}</version>
      </dependency>
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-beans</artifactId>
        <version>${spring.version}</version>
      </dependency>

      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-aspects</artifactId>
        <version>${spring.version}</version>
      </dependency>

      <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-log4j12</artifactId>
        <version>${slf4j.version}</version>
      </dependency>

    </dependencies>
  </dependencyManagement>
</project>
  • 子工程设置依赖

在子工程中的pom.xml需要设置父工程:

<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>
  <!-- 父工程 -->
  <parent>
    <groupId>cn.sm1234</groupId>
    <artifactId>parent</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <!-- 父项目的pom.xml文件的相对路径;一般可不指定 -->
    <relativePath>../parent</relativePath>
  </parent>

  <groupId>cn.sm1234</groupId>
  <artifactId>sm1234-C</artifactId>
  <version>0.0.1-SNAPSHOT</version>

  <!-- 依赖 -->
  <dependencies>

    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <!-- 版本号由父工程里面统一指定不再需要特别指定 -->
      <!-- <version>${junit.version}</version> -->
      <scope>test</scope>
    </dependency>
  </dependencies>
</project>

relativePath:父项目的pom.xml文件的相对路径。默认值为…/pom.xml。maven首先从当前构建项目开始查找父项目的pom文件,然后从本地仓库,最后从远程仓库。RelativePath允许你选择一个不同的位置;一般Eclipse找不到parent项目时可以先update project,还不行则可配置此项。

11.2.聚合

多模块项目是现在开发中比较常用的方式。

如果想一次构建多个项目模块,那则需要对多个项目模块进行聚合。

<modules>
  <module>../子项目名称1</module>
  <module>../子项目名称2</module>
  <module>../子项目名称3</module>
</modules>

12.maven搭建SSH项目实战

很实用,推荐阅读

参考:http://www.yiidian.com/maven/maven-ssh-project.html

目录
相关文章
|
4月前
|
XML 前端开发 Java
SpringMVC入门到实战------2、SpringMVC创建实例Hello SpringMVC(maven+tomcat)
这篇文章是SpringMVC框架的入门教程,详细指导了如何在IDEA中使用Maven和Tomcat创建SpringMVC工程,包括添加依赖、配置web.xml、编写控制器、创建配置文件、配置Tomcat服务器以及进行基本的测试,展示了一个简单的Hello SpringMVC示例。
SpringMVC入门到实战------2、SpringMVC创建实例Hello SpringMVC(maven+tomcat)
|
7月前
|
XML Java Shell
【深入浅出Maven开发实战】「入门教程系列」带你零基础学习和开发使用Maven开发工具实战指南(实战技术总结)(一)
【深入浅出Maven开发实战】「入门教程系列」带你零基础学习和开发使用Maven开发工具实战指南(实战技术总结)
226 1
|
7月前
|
XML Java Maven
【深入浅出Maven开发实战】「入门教程系列」带你零基础学习和开发使用Maven开发工具实战指南(实战技术总结)(二)
【深入浅出Maven开发实战】「入门教程系列」带你零基础学习和开发使用Maven开发工具实战指南(实战技术总结)
127 0
|
6月前
|
存储 搜索推荐 Java
rodert教你学Maven-实战这一篇就够了(上)
rodert教你学Maven-实战这一篇就够了
55 1
 rodert教你学Maven-实战这一篇就够了(上)
|
6月前
|
Java Linux 网络安全
在Linux上搭建Maven仓库的实战教程
在Linux上搭建Maven仓库的实战教程
384 0
|
7月前
|
Java Maven
Maven实战 Item4 -- Maven核心概念_maven junit version(1)
Maven实战 Item4 -- Maven核心概念_maven junit version(1)
|
7月前
|
前端开发 JavaScript Java
Maven实战 Item3 -- Maven项目构建2_构建一个maven2 3项目
Maven实战 Item3 -- Maven项目构建2_构建一个maven2 3项目
|
7月前
|
JavaScript 安全 前端开发
Maven实战 Item2 -- Maven项目构建(手动)_term2 配置maven(2)
Maven实战 Item2 -- Maven项目构建(手动)_term2 配置maven(2)
|
7月前
|
前端开发 Java Maven
Maven实战 Item2 -- Maven项目构建(手动)_term2 配置maven(1)
Maven实战 Item2 -- Maven项目构建(手动)_term2 配置maven(1)
|
25天前
|
XML Java 测试技术
从零开始学 Maven:简化 Java 项目的构建与管理
Maven 是一个由 Apache 软件基金会开发的项目管理和构建自动化工具。它主要用在 Java 项目中,但也可以用于其他类型的项目。
37 1
从零开始学 Maven:简化 Java 项目的构建与管理