Maven进阶
一、分模块开发与设计
分模块开发的意义
- 将原始模块按照功能拆分成若干个子模块,方便模块间的互相调用,接口共享
注意:
- 分模块开发需要先针对模块功能进行设计,再进行开发。不会先将工程开发完毕,然后进行拆分。
- 通过maven指令安装模块到本地仓库(install指令),团队内部开发需要发布模块功能到团队内部可共享的仓库中(私服)
二、依赖管理
采用分模块开发以后,我们开发的模块可以独立出来变成了一个资源,当我们需要使用的时候,就可以通过依赖的形式在其他的模块中引用了。
- 依赖指当前项目运行所需的jar,一个项目可以设置多个依赖
格式:
<!--设置当前项目所依赖的所有jar--> <dependencies> <!--设置具体的依赖--> <dependency> <!--设置所属的群组id--> <groupId>org.springframework</groupId> <!--依赖所属项目id--> <artifactId>spring-webmvc</artifactId> <!--依赖版本号--> <version>5.2.10.RELEASE</version> </dependency> </dependencies>
1、依赖传递
依赖具有传递性
直接依赖:在当前项目中通过依赖配置建立的依赖关系
间接依赖:被依赖的资源如果还依赖其他资源,那么当前项目会间接依赖其他资源
依赖传递冲突问题
- 路径优先:当依赖中出现相同的资源时,层级越深,优先级越低,层级越浅,优先级越高
- 声明优先:当资源在相同层级被依赖时,配置顺序靠前的覆盖配置顺序靠后的
- 特殊优先:当同级配置了相同资源的不同版本,后配置的覆盖先配置的
2、可选依赖与排除依赖
可选依赖指对外隐藏当前所依赖的资源,【不想把自己模块的依赖给别人用,主要用于避免依赖冲突】——不透明
<dependency> <groupId>com.itheima</groupId> <artifactId>maven_03_pojo</artifactId> <version>1.0-SNAPSHOT</version> <!--可选依赖是隐藏当前工程所依赖的资源,隐藏后对应资源将不具有依赖传递--> <optional>false</optional> </dependency>
排除依赖指主动断开依赖的资源,被排除的资源无需指定版本——不需要
<dependency> <groupId>com.itheima</groupId> <artifactId>maven_04_dao</artifactId> <version>1.0-SNAPSHOT</version> <!--排除依赖是隐藏当前资源对应的依赖关系--> <exclusions> <exclusion> <!--排除依赖资源仅指定GA即可,无需指定V--> <groupId>com.itheima</groupId> <artifactId>maven_03_pojo</artifactId> </exclusion> <exclusion> <groupId>log4j</groupId> <artifactId>log4j</artifactId> </exclusion> <exclusion> <groupId>org.mybatis</groupId> <artifactId>mybatis</artifactId> </exclusion> </exclusions> </dependency>
简单来梳理下:
A依赖B,B依赖C , C通过依赖传递会被A使用到,现在要想办法让A不去依赖C。
- 可选依赖是在B上设置<optional> , 这样A就不知道有C的存在。
- 排除依赖是在A上设置<exclusions> , A知道有C的存在,主动将其排除掉。
简而言之,自己开发的模块使用可选依赖<optional>;使用别人开发的模块只能使用排除依赖<exclusions>。
三、继承与聚合
1、聚合
- 聚合:将多个模块组织成一个整体,同时进行项目构建的过程称为聚合
- 聚合工程:通常是一个不具有业务功能的“空”工程(有且仅有一个pom文件)
- 作用:使用聚合工程可以将多个工程编组,通过对聚合工程进行构建,实现对所包含的模块进行同步构建
-
- 当工程中某个模块发生更新(变更)时,必须保障工程中与已更新模块关联的模块同步更新,此时可以使用聚合工程来解决批量模块同步构建的问题
- 分模块开发后,需要将这四个项目都安装到本地仓库,目前我们只能通过项目Maven面板的 install来安装,并且需要安装四个,如果我们的项目足够多,那么一个个安装起来还是比较麻烦的。
- 如果四个项目都已经安装成功,当ssm_pojo发生变化后,我们就得将ssm_pojo重新安装到maven仓库,但是为了确保我们对ssm_pojo的修改不会影响到其他项目模块,我们需要对所有的 模块进行重新编译,那又需要将上述的install操作再来一遍。
聚合工程的pom文件内容如下:
<?xml version="1.0" encoding="UTF-8"?> <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.itheima</groupId> <artifactId>maven_01_parent</artifactId> <version>1.0-RELEASE</version> <!--将项目的打包方式设置为pom--> <packaging>pom</packaging> <!--设置管理的模块名称,无书写顺序要求--> <modules> <module>../maven_02_ssm</module> <module>../maven_03_pojo</module> <module>../maven_04_dao</module> </modules> </project>
说明:聚合工程管理的项目在进行运行的时候,会按照模块与模块之间的依赖关系来自动决定执行的顺序,与其配置的顺序无关。
2、继承
我们已经完成了使用聚合工程去管理项目,在聚合工程中进行某一个构建操作时,被其管理的项目也会执行相同的构建操作。那么接下来,我们再来分析下,多模块开发存在的另外一个问题【重复配置】我们先来看张图:
- spring-webmvc、spring-jdbc在三个项目模块中都有出现,这样就出现了重复的内容
- spring-test只在ssm_crm和ssm_goods中出现,而在ssm_order中没有,这里是部分重复的内容
- 我们使用的spring版本目前是5.2.10.RELEASE ,假如后期要想升级spring版本,所有跟Spring相关jar包都得被修改,涉及到的项目越多,维护成本越高。
概念:继承描述的是两个工程间的关系,与java中的继承相似,子工程可以继承父工程中的配置信息,常用于依赖关系的继承。
作用:简化配置、减少版本冲突。
>> 继承关系实施步骤<<
针对父工程的pom文件的修改:
1)配置子工程中必选的依赖关系(子工程将沿用父工程中的依赖关系):
<dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>5.2.10.RELEASE</version> </dependency> ... </dependencies>
2)配置子工程中可选的依赖关系:
<dependencyManagement> <dependencies> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.16</version> </dependency> ... </dependencies> </dependencyManagement>
针对子工程的pom文件的修改:
1)在子工程中配置当前工程所继承的父工程:
<!--配置该工程的父工程--> <parent> <groupId>com.itheima</groupId> <artifactId>maven_01_parent</artifactId> <version>1.0-RELEASE</version> <!--填写父工程的pom文件--> <relativePath>../maven_01_parent/pom.xml</relativePath> </parent>
2)在子工程中配置使用父工程中可选依赖的坐标:
<dependencies> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> </dependency> </dependencies>
注意事项:
子工程中使用父工程中的可选依赖时,仅需要提供群组id和项目id,无需提供版本,版本由父工程统一提供,避免版本冲突,子工程中还可以定义父工程中没有定义的依赖关系。
3、聚合与继承的区别
作用:
聚合用于快速构建项目
继承用于快速配置
相同点:
聚合与继承的pom.xml文件打包方式为pom,可以将两种关系制作到同一个pom文件中
聚合与继承均属于设计型模块,并无实际的模块内容
不同点:
聚合是在当前模块中配置关系,聚合可以感知到参与聚合的模块有哪些
继承是在子模块中配置关系,父模块无法感知哪些子模块继承了自己
四、属性
1)定义属性
<!--定义属性--> <properties> <spring.version>5.2.10.RELEASE</spring.version> <junit.version>4.12</junit.version> </properties>
2)引用属性
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>${spring.version}</version> ... </dependency>
属性有点类似java中的变量:
此时,我们只需要更新父工程中properties标签中所维护的jar包版本,所有子项目中的版本也就跟着更新。当然除了将spring相关版本进行维护,我们可以将其他的jar包版本也进行抽取,这样就可 以对项目中所有jar包的版本进行统一维护。
五、多环境开发
我们平常都是使用自己本地的开发环境进行开发,当开发完成后,需要把开发的功能部署到测试环境供测试人员进行测试使用,等测试人员测试通过后,我们会将项目部署到生产环境上线使用。
这个时候就有一个问题,不同环境的配置不相同,不可能让三个环境都用一个数据库。
maven提供配置多种环境设定,帮助开发者使用过程中快速切换环境:
1)定义多环境
<!--配置多环境--> <profiles> <!--定义具体的环境:开发环境--> <profile> <!--定义环境对应的唯一名称--> <id>env_dep</id> <!--定义环境中专用的属性值--> <properties> <jdbc.url>jdbc:mysql://127.0.0.1:3306/ssm_db</jdbc.url> </properties> <!--设定是否为默认启动环境--> <activation> <activeByDefault>true</activeByDefault> </activation> </profile> <!--定义具体的环境:生产环境--> <profile> <id>env_pro</id> <properties> <jdbc.url>jdbc:mysql://127.2.2.2:3306/ssm_db</jdbc.url> </properties> </profile> <!--定义具体的环境:测试环境--> <profile> <id>env_test</id> <properties> <jdbc.url>jdbc:mysql://127.3.3.3:3306/ssm_db</jdbc.url> </properties> </profile> </profiles>
2)在jdbc.properties配置文件中使用${jdbc.url}
来引用该属性
3)使用方式:mvn 指令 -p 环境定义id 如:mvn install -p pro_env
注意:在SpringBoot项目的application.properties配置文件中则是使用@jdbc.url@
来引用该属性。也可以使用如下配置对资源文件开启对默认占位符${}
的解析:
<plugin> <artifactId>maven-resources-plugin</artifactId><configuration> <encoding>utf-8</encoding> <useDefaultDelimiters>true</useDefaultDelimiters></configuration> </plugin>
避坑:
在bootstrap.yml文件中@profile@无法被pom.xml中的profile配置替换问题处理,需要在pom文件中添加如下配置:
<build> ... <resources> <resource> <directory>src/main/resources</directory> <filtering>true</filtering> <includes> <include>**/**</include> </includes> </resource> </resources> ... </build>
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?