1. Java POM模块化是什么

在Java项目中,特别是在使用Maven作为构建工具时,"POM模块化"是一个重要的概念,它指的是将大型项目拆分成多个更小、更易于管理的模块(或称为子项目)。每个模块都有自己的pom.xml文件,该文件定义了模块的构建配置,包括依赖关系、插件、目标平台等。

1.1 POM(Project Object Model)

POM是Maven项目管理和构建的核心文件,它通常是一个名为pom.xml的XML文件。POM文件包含了项目的所有配置信息,Maven通过这些信息来构建项目、管理依赖以及执行其他构建任务。

1.2 模块化

模块化是一种将软件分解成一组独立但可互操作的模块的技术。在Maven项目中,模块化意味着将大型应用程序或库拆分成更小的组件,每个组件都负责一组特定的功能或业务逻辑。这些组件(即模块)可以通过Maven的依赖管理机制相互依赖,从而形成一个完整的应用程序或库。

1.3 Maven模块化项目的优点

(1)可重用性:模块可以被多个项目共享和重用。

(2)易于管理:大型项目拆分成多个小模块后,每个模块都可以独立构建和测试,从而简化了整个项目的构建和测试过程。

(3)清晰的依赖关系:通过POM文件中的依赖声明,可以清晰地看到模块之间的依赖关系。

(4)团队协作:不同的模块可以由不同的团队或开发者并行开发,提高了开发效率。

(5)灵活性:模块化使得项目更加灵活,可以更容易地添加、删除或替换模块。

1.4 Maven模块化项目的结构

一个Maven模块化项目通常包含一个父POM文件和多个子模块。父POM文件定义了所有子模块共享的构建配置和依赖管理策略。子模块则继承父POM的配置,并根据需要添加特定的配置或依赖。

1.5 示例

假设有一个名为MyProject的Maven模块化项目,它包含三个子模块:commonmodule-amodule-b。项目的目录结构可能如下所示:

MyProject/
|-- pom.xml (父POM)
|-- common/
| |-- pom.xml
| |-- src/
| |-- main/
| |-- java/
| |-- com/example/common/
|-- module-a/
| |-- pom.xml
| |-- src/
| |-- main/
| |-- java/
| |-- com/example/modulea/
|-- module-b/
|-- pom.xml
|-- src/
|-- main/
|-- java/
|-- com/example/moduleb/

在这个例子中,MyProject/pom.xml是父POM文件,它定义了所有子模块共有的配置和依赖。commonmodule-amodule-b是子模块,它们分别包含自己的pom.xml文件和源代码。这些子模块可以通过Maven的依赖机制相互依赖,也可以依赖外部库。

通过模块化,MyProject项目变得更加清晰、易于管理和维护。开发者可以独立地构建和测试每个模块,而不必担心它们之间的依赖关系。同时,模块化的结构也使得项目更加灵活,可以更容易地根据需求进行扩展或修改。

2. Java Pom两个模块需要互相引用方法示例

在Maven项目中,当两个模块(或称为子项目)需要互相引用时,通常意味着这两个模块之间存在紧密的依赖关系。然而,Maven的常规依赖管理并不直接支持循环依赖(即A依赖B,B又依赖A),因为这会导致构建过程中的死锁。不过,大多数情况下,可以通过重新设计模块结构或利用Maven的特性(如聚合和继承)来避免直接的循环依赖。

但假设我们的场景是合理的,比如两个模块分别负责不同的业务逻辑,但确实需要共享一些公共的类或接口,而这些类或接口又分布在两个模块中。这种情况下,我们可以考虑将共享的部分提取到一个新的模块中,然后让这两个模块都依赖于这个新模块。

我将展示一个简化的例子,其中两个模块module-amodule-b通过Maven的聚合(Aggregation)和继承(Inheritance)机制来组织,并假设它们通过共享一个公共的父POM来管理依赖,而不是直接互相引用。

2.1 项目结构

my-project/
|-- pom.xml (父POM)
|-- module-a/
| |-- pom.xml
| |-- src/
| |-- main/
| |-- java/
| |-- com/example/modulea/ModuleA.java
|-- module-b/
| |-- pom.xml
| |-- src/
| |-- main/
| |-- java/
| |-- com/example/moduleb/ModuleB.java
|-- common/
|-- pom.xml
|-- src/
|-- main/
|-- java/
|-- com/example/common/SharedClass.java

2.2 父POM (my-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>
<groupId>com.example</groupId>
<artifactId>my-project</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>pom</packaging> <modules>
<module>module-a</module>
<module>module-b</module>
<module>common</module>
</modules>
</project>

2.3 common 模块 (common/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>com.example</groupId>
<artifactId>my-project</artifactId>
<version>1.0-SNAPSHOT</version>
</parent> <artifactId>common</artifactId> <dependencies>
<!-- 这里可以添加common模块需要的依赖 -->
</dependencies>
</project>

2.4 module-amodule-b 的POM文件

这两个模块的POM文件将非常相似,除了它们的artifactId和可能的一些特定依赖外。它们都将依赖于common模块。

<!-- 以module-a为例 -->
<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>com.example</groupId>
<artifactId>my-project</artifactId>
<version>1.0-SNAPSHOT</version>
</parent> <artifactId>module-a</artifactId> <dependencies>
<dependency>
<groupId>com.example</groupId>
<artifactId>common</artifactId>
<version>${project.version}</version>
</dependency>
<!-- 其他依赖 -->
</dependencies>
</project>

2.5 结论

在这个例子中,module-amodule-b没有直接互相引用,而是通过共享一个common模块来避免循环依赖。这是处理Maven项目中模块间依赖关系的推荐方式。如果确实需要两个模块直接互相引用,那么可能需要重新考虑我们的项目结构或设计模式。

3. 如何使用Maven模块化

使用Maven进行模块化是一种将大型项目分解为更小、更易于管理的部分的方法。每个模块都是一个独立的Maven项目,拥有自己的pom.xml文件,但可以通过Maven的继承和聚合特性与其他模块相关联。以下是如何使用Maven进行模块化的基本步骤:

3.1 创建父POM

首先,我们需要创建一个父POM(pom.xml),它将作为所有子模块的通用配置模板。父POM通常不包含源代码,而是定义了项目共有的配置,如依赖管理、插件配置、目标平台等。

<!-- 父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.example</groupId>
<artifactId>my-project-parent</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>pom</packaging> <!-- 依赖管理 -->
<dependencyManagement>
<dependencies>
<!-- 这里定义子模块可能需要的依赖项及其版本 -->
</dependencies>
</dependencyManagement> <!-- 插件管理 -->
<build>
<pluginManagement>
<!-- 这里定义构建过程中可能需要的插件及其配置 -->
</pluginManagement>
</build> <!-- 模块列表 -->
<modules>
<module>module-a</module>
<module>module-b</module>
<!-- 其他子模块 -->
</modules>
</project>

注意:<packaging>pom</packaging>表明这是一个聚合POM,它不会构建任何实际的产品,而是用来聚合和管理其他模块。

3.2 创建子模块

然后,我们需要在父POM的同级目录下(或指定的任何子目录中)创建子模块。每个子模块都应该有自己的pom.xml文件,并且通常会继承自父POM。

<!-- 子模块A的POM (位于module-a目录) -->
<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>com.example</groupId>
<artifactId>my-project-parent</artifactId>
<version>1.0-SNAPSHOT</version>
</parent> <artifactId>module-a</artifactId> <!-- 依赖项(如果需要的话) -->
<dependencies>
<!-- 这里可以声明具体的依赖项,版本号可以从父POM中继承 -->
</dependencies> <!-- 其他配置 -->
</project>

3.3 构建项目

在父POM所在的目录下运行Maven命令,Maven会自动找到并构建所有列在<modules>标签下的子模块。

bash复制代码

mvn clean install

这个命令会首先清理之前构建生成的文件,然后编译、测试并安装所有子模块到本地Maven仓库中。

3.4 依赖管理

在父POM中定义的<dependencyManagement>部分允许我们指定依赖项及其版本号,但不会在父POM中实际引入这些依赖项。子模块可以通过声明相同的依赖项(不包括版本号)来继承这些依赖项及其版本号。

3.5 插件管理

类似地,<pluginManagement>部分允许我们在父POM中定义插件及其配置,但不会在父POM中实际执行这些插件。子模块可以通过继承这些插件配置来简化插件配置过程。

3.6 结论

通过Maven模块化,我们可以将大型项目分解为更小、更易于管理的部分,同时利用Maven的继承和聚合特性来共享配置和依赖项。这有助于提高项目的可维护性、可重用性和可扩展性。

Java Pom两个模块需要互相引用怎么办的更多相关文章

  1. 关于node中两个模块相互引用却不会死循环的问题

    关于node中两个模块相互引用却不会死循环的问题 node中是通过require来导入加载模块的,require有两个作用: 1.加载文件模块并执行里面的代码 2.拿到被加载文件模块导出的接口对象 现 ...

  2. eclipse在多modules项目结构下避免模块间依赖引用的场景

    这个在单一classLoader时,不会有问题.如果多classloader下会有问题. 假设工程有两个模块,module2 依赖module1 当我们执行mvc eclipse:eclipse后,然 ...

  3. OSGi 系列(一)之什么是 OSGi :Java 语言的动态模块系统

    OSGi 系列(一)之什么是 OSGi :Java 语言的动态模块系统 OSGi 的核心:模块化.动态.基于 OSGi 就可以模块化的开发 java 应用,模块化的部署 java 应用,还可以动态管理 ...

  4. 在pom包中添加spring-boot-starter-test包引用

    有很多网友会时不时的问我,spring boot项目如何测试,如何部署,在生产中有什么好的部署方案吗?这篇文章就来介绍一下spring boot 如何开发.调试.打包到最后的投产上线. 开发阶段 单元 ...

  5. java中两种类型变量

    Java中有两种类型的变量,一种是对象类型,另一种是基础类型(primitive type). 对象类型普遍采用引用的方式,比如 List a = new ArrayList(); List b = ...

  6. Java中带包(创建及引用)的类的编译

    Java中带包(创建及引用)的类的编译与调试 java源程序的编译大家都知道,也就是cmd中到源文件所在目录下javac **.java即可,当程序中有包声明还能简简单单的直接javac **.jav ...

  7. OSGi是什么:Java语言的动态模块系统(一)

    OSGi是什么 OSGi亦称做Java语言的动态模块系统,它为模块化应用的开发定义了一个基础架构.OSGi容器已有多家开源实现,比如Knoflerfish.Equinox和Apache的Felix.您 ...

  8. [转] 有关java中两个整数的交换问题

    转载申明:本文主要是用于自己学习使用,为了完善自己的只是框架,没有任何的商业目的. 原文来源:有关Java中两个整数的交换问题 如果侵权,麻烦告之,立刻删除. 在程序开发的过程,要交换两个变量的内容, ...

  9. java 传参方式--值传递还是引用传递

    java 传参方式--值传递还是引用传递 参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递.写它是为了揭穿普遍存在的一种神话,即认为 Java 应用程序按引用 ...

  10. Java中到底是值传递还是引用传递?

    Java中到底是值传递还是引用传递? 我们先回顾一下基本概念 实参和形参 参数在编程语言中是执行程序需要的数据,这个数据一般保存在变量中.在Java中定义一个方法时,可以定义一些参数, 举个例子: p ...

随机推荐

  1. iOS开发基础133-崩溃预防

    现代移动应用的用户体验依赖于其稳定性和可靠性.然而,在开发过程中,我们时常会遇到各种崩溃问题.崩溃不仅会影响用户的使用体验,还可能损害应用的声誉.因此,本文将详细介绍一个名为CrashPreventi ...

  2. 深入理解 Vue 3 组件通信

    在 Vue 3 中,组件通信是一个关键的概念,它允许我们在组件之间传递数据和事件.本文将介绍几种常见的 Vue 3 组件通信方法,包括 props.emits.provide 和 inject.事件总 ...

  3. java面试一日一题:在创建微服务时,是用RPC还是http

    问题:请讲下在做微服务时,是使用RPC还是http 分析:该问题主要考察对RCP及http的理解,也关系到在进行微服务选型时的两大方向,dubbo和springCloud,都是RPC框架,但前者是RP ...

  4. Python 在PDF中添加、替换、或删除图片

    PDF文件中的图片可以丰富文档内容,提升用户的阅读体验.除了在PDF中添加图片外,有时也需要替换或删除其中的图片,以改进视觉效果或更新信息.本文将提供以下三个示例,介绍如何使用Python 操作PDF ...

  5. 如何理解IOC中的“反转”和DI中的“注入”

    在理解 IOC 中的"反转"和 DI 中的"注入"之前,首先要理解原本的控制流程. 在传统的应用程序中,对象之间的依赖关系通常由调用方(例如客户端或者上层模块) ...

  6. Jmeter函数助手6-time

    time函数用于获取不同格式的当前时间(年月日时分秒). Format string for SimpleDateFormat (optional):时间格式,填入如yyyyMMdd-HHmmss.d ...

  7. 【Vue】分组类型排名查询功能

    一.书接上回: https://www.cnblogs.com/mindzone/p/17749725.html 这个效果被否决了,产品要求一定要UI的来,UI效果如图: 按人为主体的时候,固定有4个 ...

  8. 【Vue2】Computed 计算属性

    计算属性在编写的时候是一个方法 但是在调用的时候作为属性使用 <!DOCTYPE html> <html lang="en"> <head> & ...

  9. 电脑时间不同步导致的上网报错:core/proxy/vmess/encoding: failed to read response header > websocket: close 1006 (abnormal closure): unexpected EOF

    报错内容: 2023/12/16 14:08:56 [Warning] [775541588] xxxxx.com/core/app/proxyman/outbound: failed to proc ...

  10. 【转载】 Ubuntu下使用VSCode的launch.json及tasks.json编写

    版权声明:本文为CSDN博主「子木呀」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明.原文链接:https://blog.csdn.net/qq_41687938/a ...