【Maven四】——maven聚合和继承

news2024/11/24 21:06:39

系列文章目录

Maven之POM介绍
maven命令上传jar包到nexus
【Maven二】——maven仓库
【Maven三】——maven生命周期和插件


聚合和继承

  • 系列文章目录
  • 前言
  • 一、什么是maven的聚合和继承&why
  • 二、聚合
  • 三、继承
    • 1.可继承的POM元素
    • 2.依赖管理
    • 3.插件管理
  • 四、聚合与继承的关系
  • 五、约定优于配置
  • 六、反应堆
    • 1.反应堆的构件顺序
  • 七、裁剪反应堆
  • 总结


前言

由于在具体项目开发过程中对于maven的理解和掌握处于基本运用的阶段,了解maven过于片面,所以本篇博客是博主学习《maven实战》书籍之后对maven聚合和继承的总结,绝大多数内容源于《maven》实战这本书籍。
在这里插入图片描述


一、什么是maven的聚合和继承&why

随着技术飞速发展,各类用户对软件的要求越来越高,软件也变得越来越复杂。
软件设计人员往往会采用各种方式对软件划分模块,已得到更加清晰的设计及更高的复用性。

当把Maven应用到实际项目中的时候,也需要将项目分成不同的模块。
Maven的聚合特性能够把项目的各个模块聚合在一起构建,而maven的继承特性则能帮助抽取各个模块相同的依赖和插件等配置。

二、聚合

例如:现在有一个注册服务下面有两个模块,account-email和account-persist。如果我们想要一次构建两个项目(模块),而不是到两个模块的目录下分别执行mvn命令。

为了能够一次构建account-email和account-persist两个模块,我们需要格外的名为account-aggregator的模块,然后通过该模块构建整个项目的所有模块。

account-aggregator作为一个maven项目,它也必须有它自己的POM。
下面是account-aggregator的pom.xml内容:
在这里插入图片描述
上面的POM依旧使用了账户注册服务共同的groupId com.juvenxu.mvnbook.account,artifactId为独立的account-aggregator,版本也与其他两个模块一致。
**但是这里的packaging,值为POM。**而account-email和account-persist的packaging为jar。对于聚合模块来说,其打包方式packaging的值为POM,否则就无法构建。

而modules这是实现聚合的最核心的配置。用户可以通过在一个打包方式为pom的Maven项目中声明任意数量的module元素来实现模块的聚合。

这里每一个module的值都是一个当前POM的相对目录。例如:account-aggregator的POM的路径为D:…\code\account-aggregator\pom.xml那么account-email就对应了目录D:…\code\account-aggregator\account-email\。这两个目录各自包含了pom.xml、src/main/java/、src/test/java等内容。离开account-aggregator也能够独立构建。

一般来说,为了方便快速定位内容,模块所处的目录名称应当与其artifactId一致,不过这不是maven的要求,用户也可以将account-email项目放到email-account/目录下。
这是聚合的配置就需要相应地改成email-account

为了方便用户构建项目、通常将聚合模块放在项目目录的最顶层,其他模块作为聚合模块的子目录存在,这样当用户得到源码的时候,第一眼发现的就是聚合模块的POM不用从多个模块中去寻找聚合模块来构建整个项目。
如图:
在这里插入图片描述
当然也可以是平行的目录结构
在这里插入图片描述
如果使用平行目录,聚合模块的POM也需要做相应的修改

<modules>
<module>../account-email</module>
</modules>

从聚合模块运行mvn clean install命令得到的输出
在这里插入图片描述
maven首先解析聚合模块的POM,分析要构建的模块、并计算出一个反应堆构建顺序,探后根据这个顺序依次构建各个模块。

三、继承

面向对象设计中,可以建立一种类的父子结构,然后父类中声明一些字段和方法供子类继承。这样可以做到“一处声明,多处使用”。
类似的在maven世界中,也有类似的机制能让我们抽取出重复的配置。做到“一处声明,多处使用”这就是POM的继承。

我们创建POM的父子结构,然后在父POM中声明一些配置供子POM继承。
继续以账户注册服务为基础。在account-aggregator下创建一个名为account-parent的子目录,然后在该子目录下建立一个所有出account-aggregator之外的模块的父模块。
pom.xml文件如下:
在这里插入图片描述
需要注意的一点,它的packaging为pom,这一点与聚合模块一样,作为父模块的POM,其打包类型也必须为pom,由于父模块只是为了帮助消除配置的重复,因此它本身不包含POM之外的项目文件,也不需要src/maini/java之类的文件夹。

有了父模块,来看看继承父模块的子模块。将account-meail的POM修改如下:
在这里插入图片描述
上述POM中使用parent元素声明父模块快,parent下的元素groupId、artifactId和version指定了父模块的坐标,元素relativePath表示父模块POM的相对路径,该例中的…/account-parent/pom.xml表示父POM的位置在与account-email/目录平行的account-parent/目录下。
当项目构建时,maven会首先更具relativePath检查父POM,如果找不到,再从本地仓库中查找。relativePath的默认值为…/pom.xml,也就是说,maven默认父POM在上一层目录下。

1.可继承的POM元素

groupId:项目组ID
version:项目版本
description:项目的描述信息
organization:项目的组织信息
inceptionYear:项目的创始年份
url:项目的URL地址
developers:项目的开发者信息
contribution:项目的贡献者信息
distributionManagement:项目的部署配置
issueManagement:项目的缺陷跟踪系统信息
ciManagement:项目的持续继承系统信息
scm:项目的版本控制系统信息
mailingLists:项目的邮件列表信息
properties:自定义的Maven属性
dependencies:项目的依赖配置
dependencyManagement:项目的依赖管理配置
repositories:项目的仓库配置
build:包括项目的硬盘吗目录配置、输出目录配置、插件配置、插件管理配置等。
reporting:包括项目的报告输出目录配置、报告插件配置等。

2.依赖管理

可继承元素列表包含了dependencies元素,说明依赖是会被继承的。在子模块account-email和account-persist同时依赖了org.springframework:spring-core:2.5.6、org.springframework:spring-beans:2.5.6、org.springframework:spring-context:2.5.6和Junit:junit4.7等等,因此可以将这些依赖配置放到父模块account-parent中,两个子模块就能够移除这些依赖,简化配置。

但是这样会存在一些问题,如果将来项目需要加入一个account-util模块,该模块只是提供一些简单的帮助工具,与springframework完全无关,难道也让他依赖spring-core、spring-beans和spring-context吗?

maven提供的dependencyManagement元素既能让子模块继承到父模块的依赖配置,又能保证子模块依赖使用的灵活性。在dependencyManagement元素下的依赖声明不会引入实际的依赖,不过它能够约束dependencies下的依赖使用。例如:可以在account-parent中加入这样的dependencyManagement配置如:
在这里插入图片描述
首先父POM将springframework和junit依赖的版本以Maven变量的形式提取了出来,不仅消除了一些重复也使得各依赖的版本处于更加明显的位置。
这是使用的dependencyManagement声明的依赖既不会给account-parent引入依赖,也不会给它的子模块引入一依赖。不过这段配置是会被继承的。

现在修改account-email的POM如:

在这里插入图片描述
上述的POM中的依赖配置较原来简单了一些,所有的springframework依赖只配置了groupId和artifactId,省去了version,而Junit依赖不仅省去了version,还省去了依赖范围scope。这些信息可以省略是因为account-email继承了account-parent中的dependencyManagement配置。完整的依赖声明已经包含在父POM中,子模块只需要配置简单的groupId和artifactId就能获得对应的依赖信息,从而引入正确的依赖。

使用这种依赖管理机制似乎不能减少太多的POM配置,但强烈推荐采用这种方法。
主要原因是在于父POM中声明之后,子模块在使用依赖的使用就无须声明版本,也就不会发生多个子模块使用依赖版本不一致的情况。这可以降低依赖冲突的几率。
如果子模块不生命依赖的使用,即使依赖已经在父POM的dependencyManagement中声明了,也不会产生任何实际的效果。

补充依赖范围为import的依赖范围,该依赖范围只在dependencyManagement元素下才有效果,使用该范围的依赖通常指向一个POM,作用是将目标POM的dependencyManagement配置导入并合并到当前POM的dependencyManagement元素中。例如想要在另一个模块中使用
与上面dependencyManagement的一样的配置,除了赋值配置或者继承这两种方式之外,还可以使用import范围依赖,将这一配置导入。如:
在这里插入图片描述

3.插件管理

maven也提供了pluginManagement元素帮助管理插件。该元素中配置的依赖不会造成实际的插件调用行为,当POM中配置了真正的plugin元素,并且其groupId和artifactId与pluginManagement中配置的插件匹配时,pluginManagement的配置才会影响实际的插件行为。

例如:
配置了maven-source-plugin,将其jar-no-fork目标绑定到了verity生命周期阶段,以生成项目源码包。如果一个项目中有很多子模块,并且需要得到所有这些模块的源码包,那么更好的方法是父POM中使用pluginManagement配置插件。
如:
在这里插入图片描述
当子模块需要生成源码包的时候,只需要如下配置:
在这里插入图片描述
子模块声明使用了maven-source-plugin插件,同时又继承了父模块的pluginManagement配置。如果子模块不需要使用父模块中pluginManagement,可以尽管将其忽略,如果子模块需要不同的插件配置,则可以执行配置以覆盖父模块的pluginManagement配置。

四、聚合与继承的关系

多模块maven项目中聚合与继承其实是两个概念,其目的完全不同。前者主要是为了方便快速构建项目,后者主要是为了消除重复配置。

对于聚合模块来说,它知道有哪些被聚合的模块,但那些被聚合的模块不知道这个聚合模块的存在。
对于继承关系的父POM来说,他不知道有哪些子模块继承于它,但那些子模块都必须知道自己的父POM是什么。

共同点:
聚合POM与继承关系的父POM的packaging都必须是pom,同时聚合模块与继承关系的父模块除了POM之外都没有实际内容。

在这里插入图片描述
在现有的实际项目中,往往一个POM即是聚合POM,又是父POM。

五、约定优于配置

标准的重要性已经不用过于强调,想象一下,如果不是所有程序员都是基于HTTP协议开发Web应用,互联网会乱成什么样。各个版本的IE,FireFox等浏览器之间的差异已经让很多开发者头疼不已。
java成功的重要原因之一就是它能够屏蔽大部分操作系统的差异,XML流行的原因之一是所有语言都接受它。而maven提倡的“约定优于配置”这是maven最为核心的设计理念之一。
那为什么要使用约定而不是自己更灵活的配置呢?原因之一是使用约定可以大量减少配置。

六、反应堆

在一个多模块的Maven项目中,反应堆(Reactor)是指所有模块组成的一个构件结构。对于单模块的项目,反应堆就是该模块本身。对于多模块项目来说,反应堆包含了模块之间的依赖和继承关系,从而能够自动计算出合理的模块构件顺序。

1.反应堆的构件顺序

如:account-aggregator的聚合配置如下:
在这里插入图片描述
构建完成之后会有这样的输出:

在这里插入图片描述
可以看出反应堆的构建顺序,依次为account-aggregator、account-parent、account-email、account-persist。
在这里插入图片描述
从上至下的箭头表示POM的读取次序,但这不足以决定反应堆的构建顺序,maven还需要考虑模块之间的继承和依赖关系。图中有向虚线表示模块之间的继承或者依赖关系。上面account-email、account-persit依赖与account-parent,那么account-parent就必须先于另外两个模块构建。

实际的构建顺序是:
maven按顺序读取POM,如果该POM没有依赖模块,那么就构建该模块,否则就先构建其依赖模块,如果该依赖模块还依赖于其他模块,则进一步限购键依赖的依赖。
模块间的依赖关系会将反应堆构成一个有向非循环图,各个模块是该图的节点,依赖关系构成有向边。这个图不允许出现循环,因此,当出现模块A依赖与B,而B依赖于A的情况时,maven会报错。

七、裁剪反应堆

有时会需要仅仅构建完整反应堆中的某些个模块,maven提供很多的命令行选项支持裁剪反应堆,输出mvn-h可以看到这些选项。


总结

清楚了maven项目的聚合和继承之后,完全能够搭建起多模块的Maven项目与简洁的管理maven的依赖。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/772010.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

java电子病历系统源码

电子病历系统采取结构化与自由式录入的新模式&#xff0c;自由书写&#xff0c;轻松录入。化实现病人医疗记录&#xff08;包含有首页、病程记录、检查检验结果、医嘱、手术记录、护理记录等等。&#xff09;的保存、管理、传输和重现&#xff0c;取代手写纸张病历。不仅实现了…

【Docker】Docker安装与操作

docker的安装与命令 一、安装 docker1. 安装依赖包2. 信息查看 二、Docker 镜像操作1. 搜索镜像2. 获取镜像3. 镜像加速下载4. 查看镜像相关信息5. 删除镜像6. 上传镜像7. 存出和载入镜像 三、Docker 容器操作1. 创建容器2. 查看容器3. 启动容器4. 停止容器5. 进入容器6. 容器与…

四、DML-4.小结

一、数据记录操作 1、添加数据 给指定字段添加数据 insert into employee(id, workno, name, gender, age, idcard,entrydate) values (1, 001,Itcast, 男, 18, 123456789012345678, 2023-12-12); INSERT INTO 表名 (字段名1, 字段名2, ...) VALUES (值1, 值2, ...); 给全部…

Linux--使用者管理(job control)

Linux–使用者管理(job control) 文章目录 Linux--使用者管理(job control)前言一、任务管理(job control)二、&三、 将目前的任务丢到后台中暂停 -- ctrlz四、jobs -- 查看目前的后台任务状态五、fg -- 将后台任务拿到前台来处理六、bg -- 让任务在后台下的状态变为运行中…

LeetCode 1493. 删掉一个元素以后全为 1 的最长子数组 - 二分 + 滑动窗口

删掉一个元素以后全为 1 的最长子数组 提示 中等 90 相关企业 给你一个二进制数组 nums &#xff0c;你需要从中删掉一个元素。 请你在删掉元素的结果数组中&#xff0c;返回最长的且只包含 1 的非空子数组的长度。 如果不存在这样的子数组&#xff0c;请返回 0 。 提示 1&a…

2023-07-18力扣今日二题-太难了吧

链接&#xff1a; LCP 75. 传送卷轴 题意&#xff1a; 给一个正方形迷宫&#xff0c;主角是A&#xff0c;每次可以上下左右走一格子&#xff0c;有四种类型的格子&#xff1a;墙、初始位置、魔法水晶、空地 另一个人B&#xff0c;可以传送一次A&#xff0c;只能在空地传送&…

Twisted Circuit

题目描述 输入格式 The input consists of four lines, each line containing a single digit 0 or 1. 输出格式 Output a single digit, 0 or 1. 题意翻译 读入四个整数 00 或者 11&#xff0c;作为如图所示的电路图的输入。请输出按照电路图运算后的结果。 感谢PC_DOS …

算法与数据结构-排序

文章目录 一、如何分析一个排序算法1.1 排序算法的执行效率1.1.1 最好情况、最坏情况、平均情况时间复杂度1.1.1.1 最好、最坏情况分析1.1.1.2 平均情况分析 1.1.2 时间复杂度的系数、常数 、低阶1.1.3 比较次数和交换&#xff08;或移动&#xff09;次数 1.2 排序算法的内存消…

第六届字节跳动青训营报录比(宣传大使)

统计 前端基础卷&#xff1a;105 前端基础班&#xff1a;120-22(笔试不过基础班&#xff0c;宣传大使奖励进入&#xff09;98 前端进阶卷&#xff1a;77 前端进阶班&#xff1a;18-216 后端基础卷&#xff1a;151 后端基础班&#xff1a;220 后端进阶卷&#xff1a;133 后端进…

【论文笔记】KDD2019 | KGAT: Knowledge Graph Attention Network for Recommendation

Abstract 为了更好的推荐&#xff0c;不仅要对user-item交互进行建模&#xff0c;还要将关系信息考虑进来 传统方法因子分解机将每个交互都当作一个独立的实例&#xff0c;但是忽略了item之间的关系&#xff08;eg&#xff1a;一部电影的导演也是另一部电影的演员&#xff09…

133、仿真-基于51单片机太阳能热水器水温水位智能监控仪报警设计(Proteus仿真+程序+配套资料等)

方案选择 单片机的选择 方案一&#xff1a;STM32系列单片机控制&#xff0c;该型号单片机为LQFP44封装&#xff0c;内部资源足够用于本次设计。STM32F103系列芯片最高工作频率可达72MHZ&#xff0c;在存储器的01等等待周期仿真时可达到1.25Mip/MHZ(Dhrystone2.1)。内部128k字节…

【QT】元对象系统学习笔记(二)

QT元对象系统 01、属性系统1.1、 属性基础1.2、 QVariant类1.3、 使用QObject类存取属性值与动态属性1.4、 使用反射机制获取属性信息 02、信号与槽2.1、 信号与槽原理2.2、 创建信号与槽2.3、 信号与槽的连接2.4、 断开信号与槽2.5、 关键字原型 03、对象树与生命期3.1、组合模…

【TiDB理论知识 03】TiKV-持久化与数据读取

目录 一 TiKV架构和作用 二 数据持久化 1 RocksDB&#xff1a;写入 写入过程 第一步 &#xff1a;WAL 写日志 &#xff08;磁盘日志&#xff09; 第二步&#xff1a;写MemTable (内存中) 第三步 &#xff1a; 转存为immutable MemTable&#xff08;内存中&#xff09; …

JAVA集成国密SM4

JAVA集成国密SM4加解密 一、pom配置二、代码集成2.1、目录结构2.2、源码2.3、测试 三、遇到的坑3.1、秘钥长度3.2、转码问题 四、相关链接 国密算法概述&#xff1a;https://blog.csdn.net/qq_38254635/article/details/131801527 SM4对称算法 SM4 无线局域网标准的分组数据算…

线性表的链式存储结构以及顺序表和链表的比较

2.2线性表的链式存储结构 **链式存储结构&#xff1a;**结点在存储器中的位置是任意的&#xff0c;即逻辑上相邻的数据元素在物理上不一定相邻。 线性表的链式表示又称为非顺序映像或链式映像 这组存储单元既可以是连续的&#xff0c;也可以是不连续的&#xff0c;甚至是零散…

C++基础算法前缀和和差分篇

&#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;C算法 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&#x1f349;留言 主要讲解了前缀和和差分算法 文章目录 Ⅳ. 前缀和 和 差分Ⅵ .Ⅰ前缀和…

AtcoderABC244场

A - Last LetterA - Last Letter 题目大意 给定一个长度为N的字符串S&#xff0c;由小写英文字母组成&#xff0c;打印出S的最后一个字符。 思路分析 题目要求打印出字符串S的最后一个字符&#xff0c;可以直接通过访问S的最后一个元素来获取该字符。可以使用字符串的back()…

旅游卡小程序软件招商加盟代理

旅游卡小程序软件招商加盟代理 我国人民生活水平的提高&#xff0c;旅游业成为了人们生活中必不可少的一部分。旅游卡小程序软件作为旅游行业的重要组成部分&#xff0c;也日益受到人们的关注。如今&#xff0c;旅游卡小程序软件招商加盟代理已经成为了一个热门的投资创业项…

$.getScript()方法获取js文件

通过$.getScript(‘xxxx.js’)获取xxxx.js文件&#xff0c;这时的ajax是一个get请求的状态&#xff0c;如果进行了入参data的赋值那么他就会跟在url后面,同理获取json文件&#xff0c;css文件。 一开始没想起这茬。。。

使用PostgreSQL创建高级搜索引擎

​本文我们将探索PostgreSQL中的全文搜索功能&#xff0c;并研究我们能够复制多少典型搜索引擎功能。 如果您想跟随并尝试示例查询&#xff08;我们建议这样做&#xff0c;这样更有趣&#xff09;&#xff0c;可以使用来自Kaggle的Wikipedia电影情节数据集执行代码示例。要导入…