Java之模块化详解
Java模块化,作为Java 9引入的一项重大特性,通过Java Platform Module System (JPMS) 实现,为Java开发者提供了更高级别的封装和依赖管理机制。这一特性旨在解决Java应用的封装性、可维护性和性能问题,使得开发者能够构建更加结构化和维护性更强的应用程序。
一、Java模块化的概念与特点
1、模块的定义与结构
在Java模块化中,一个模块是一组相互关联的包和资源,以及一个模块描述符(module-info.java)。模块描述符定义了模块的名称、它所需的其他模块,以及它要向外导出的包。每个模块都包含一个module-info.java文件,这个文件声明了模块的名称、模块间的依赖关系、导出的包、开放的包、提供的服务以及使用的服务。
- 模块名称:
- 模块名称在项目中必须唯一,用于标识模块。
- 依赖关系:
- 模块可以声明对其他模块的依赖关系,这些依赖关系在编译和运行时是必需的。
- 导出的包:
- 模块可以声明哪些包是对外可见的,即可以被其他模块访问。
- 开放的包:
- 模块可以声明哪些包是可以通过反射访问的,即使这些包不是对外可见的。
- 提供的服务:
- 模块可以声明它提供的服务接口,其他模块可以使用这些服务。
- 使用的服务:
- 模块可以声明它使用的服务接口,这些服务由其他模块提供。
2、 模块化的特点
- 强封装性:
- 模块化强制要求定义模块间的显式依赖,以及公开的API。这有效地隐藏了实现细节,减少了耦合。
- 显式依赖:
- 模块必须显式声明它们依赖的其他模块,这提高了系统的可维护性和透明度。
- 性能提升:
- 模块化有助于JVM在加载和验证类时更高效,因为它可以跳过对非必需模块的处理。
- 安全性增强:
- 由于模块化限制了对非公开API的访问,这能够降低安全风险。
- 更易维护:
- 模块化对项目的逻辑和物理分割有助于更好的维护和快速迭代。
二、Java模块化的应用场景
Java模块化适用于各种规模的项目,但尤其适用于大型应用程序和库的开发。通过模块化,开发者可以:
1、更好地构造和维护大型应用程序和库:
- 模块化使得项目结构更加清晰,便于理解和维护。
2、 提高系统的可重用性和可扩展性:
- 模块是独立的、可重用的部分,可以轻松地集成到其他项目中。
3、解决“JAR地狱”问题:
- 显式的模块依赖关系避免了类路径冲突和版本冲突。
4、 优化系统性能:
- 模块化可以减少应用的启动时间和运行时内存占用。
三、Java模块化的代码示例
下面是一个简单的Java模块化代码示例,展示了如何在实际项目中使用模块化。
1、创建模块目录和文件
首先,我们创建一个新的Java项目,并设置为Java 9或更高版本的模块化项目。在项目中,我们创建两个模块:com.example.app
和com.example.utils
。
项目结构如下:
src/
├── com.example.app/
│ ├── module-info.java
│ └── com/
│ └── example/
│ └── app/
│ └── Main.java
└── com.example.utils/├── module-info.java└── com/└── example/└── utils/└── StringUtils.java
2、编写模块描述符
在每个模块的根目录下,我们创建一个module-info.java文件,定义模块的名称、依赖关系和导出的包。
com.example.utils
的module-info.java文件:
module com.example.utils {exports com.example.utils;
}
com.example.app
的module-info.java文件:
module com.example.app {requires com.example.utils;
}
3、 实现模块功能
在com.example.utils
模块中,我们创建一个StringUtils
类,提供一个将字符串转换为大写的方法。
com.example.utils.StringUtils
类:
package com.example.utils;public class StringUtils {public static String capitalize(String input) {return input.toUpperCase();}
}
在com.example.app
模块中,我们创建一个Main
类,使用com.example.utils
模块中的StringUtils
类。
com.example.app.Main
类:
package com.example.app;import com.example.utils.StringUtils;public class Main {public static void main(String[] args) {String message = StringUtils.capitalize("hello world");System.out.println(message);}
}
4、 编译和运行模块化项目
使用javac
命令编译模块化项目:
javac -d mods --module-source-path src $(find src -name "*.java")
使用java
命令运行模块化项目:
java --module-path mods -m com.example.app/com.example.app.Main
运行结果:
HELLO WORLD
四、模块化在Java开发中的优势和重要性
1、 提高封装性和安全性
模块化通过强制要求定义模块间的显式依赖和公开的API,有效地隐藏了实现细节,减少了耦合。同时,由于模块化限制了对非公开API的访问,这能够降低安全风险。这对于构建大型应用程序和库尤为重要,因为它们通常包含许多相互依赖的组件,需要严格的封装和访问控制。
2、 简化依赖管理
在Java模块化之前,依赖管理通常是通过类路径(classpath)来实现的。然而,类路径存在一些问题,如“JAR地狱”问题(即多个JAR文件包含相同的类名,导致冲突)和版本冲突问题。模块化通过显式的模块依赖关系避免了这些问题,使得依赖管理更加简单和准确。
3、 优化系统性能
模块化有助于JVM在加载和验证类时更高效,因为它可以跳过对非必需模块的处理。这意味着在启动时间和运行时内存占用方面,模块化应用程序通常比非模块化应用程序表现更好。这对于构建需要快速启动和响应的应用程序尤为重要。
4、 易于维护
模块化应用的结构更清晰,便于理解和维护。每个模块的职责明确,易于独立更新和测试。这使得大型应用程序和库的开发更加高效和可靠。同时,模块化还促进了代码的重用和共享,因为模块可以作为独立的组件进行开发和部署。
5、 促进微服务架构
随着微服务架构的流行,模块化在Java开发中的重要性日益凸显。微服务架构强调将应用程序拆分成一系列小的、自治的服务。这些服务可以独立开发、部署和扩展。模块化提供了实现微服务架构的理想工具,因为它允许开发者将应用程序拆分成独立的、可重用的模块。
五、如何在实际项目中应用模块化
要在实际项目中应用模块化,需要遵循以下步骤:
1、分析应用程序结构
首先,需要仔细分析应用程序的结构,识别逻辑分组来创建模块。这通常涉及将相关的功能和资源组织在一起,形成一个独立的模块。
2、创建模块目录和文件
为每个模块创建一个独立的目录和文件结构。在模块的根目录下创建module-info.java文件,定义模块的名称、依赖关系和导出的包。
3、 编写模块描述符
在module-info.java文件中,声明模块的名称、它所需的其他模块以及它要向外导出的包。确保依赖关系是显式的,并且导出的包是必需的。
4、 实现模块功能
在模块中编写代码,实现所需的功能。确保代码遵循模块化的规则和约束,如封装性、依赖性和可见性。
5、 编译和测试模块
使用javac
命令编译模块化项目,并确保没有编译错误。然后,编写测试用例来验证模块的功能和依赖关系是否正确。
6、 部署和运行模块化应用程序
将编译后的模块打包并部署到目标环境中。使用java
命令运行模块化应用程序,并验证其性能和行为是否符合预期。
7、 持续优化和维护
模块化应用程序的维护和更新更加容易,因为每个模块都是独立的、可重用的部分。随着应用程序的发展,需要持续优化和维护模块化结构,以确保其适应不断变化的需求和环境。
六、总结
Java模块化是一项强大的特性,它为Java开发者提供了更好的封装性、更清晰的项目结构和更强大的依赖管理功能。通过模块化,开发者可以构建更加结构化和维护性更强的应用程序,提高系统的可重用性和可扩展性。同时,模块化还有助于解决“JAR地狱”问题、优化系统性能和提高安全性。在实际项目中应用模块化需要仔细分析应用程序结构、创建模块目录和文件、编写模块描述符、实现模块功能、编译和测试模块、部署和运行模块化应用程序以及持续优化和维护。通过这些步骤,开发者可以充分利用模块化的优势来构建高质量的Java应用程序。
相关文章:

Java之模块化详解
Java模块化,作为Java 9引入的一项重大特性,通过Java Platform Module System (JPMS) 实现,为Java开发者提供了更高级别的封装和依赖管理机制。这一特性旨在解决Java应用的封装性、可维护性和性能问题,使得开发者能够构建更加结构化…...

HTB:Knife[WriteUP]
目录 连接至HTB服务器并启动靶机 1.How many TCP ports are open on Knife? 2.What version of PHP is running on the webserver? 并没有我们需要的信息,接着使用浏览器访问靶机80端口 尝试使用ffuf对靶机Web进行一下目录FUZZ 使用curl访问该文件获取HTTP头…...

MOE论文详解(4)-GLaM
2022年google在GShard之后发表另一篇跟MoE相关的paper, 论文名为GLaM (Generalist Language Model), 最大的GLaM模型有1.2 trillion参数, 比GPT-3大7倍, 但成本只有GPT-3的1/3, 同时效果也超过GPT-3. 以下是两者的对比: 跟之前模型对比如下, 跟GShard和Switch-C相比, GLaM是第一…...

LeetCode322:零钱兑换
题目链接:322. 零钱兑换 - 力扣(LeetCode) 代码如下 class Solution { public:int coinChange(vector<int>& coins, int amount) {vector<int> dp(amount 1, INT_MAX);dp[0] 0;for(int i 0; i < coins.size(); i){fo…...

速盾:高防 cdn 提供 cc 防护?
在当今网络环境中,网站面临着各种安全威胁,其中 CC(Challenge Collapsar)攻击是一种常见的分布式拒绝服务攻击方式。高防 CDN(Content Delivery Network,内容分发网络)作为一种有效的网络安全防…...

【大数据应用开发】2023年全国职业院校技能大赛赛题第10套
如有需要备赛资料和远程培训,可私博主,详细了解 目录 任务A:大数据平台搭建(容器环境)(15分) 任务B:离线数据处理(25分) 任务C:数据挖掘(10分) 任务D:数据采集与实时计算(20分) 任务E:数据可视化(15分) 任务F:综合分析(10分) 任务A:大数据平台搭…...

【源码部署】解决SpringBoot无法加载yml文件配置,总是使用8080端口方案
打开idea,file ->Project Structure 找到Modules ,在右侧找到resource目录,是否指定了resource,点击对应文件夹会有提示...

2010年国赛高教杯数学建模B题上海世博会影响力的定量评估解题全过程文档及程序
2010年国赛高教杯数学建模 B题 上海世博会影响力的定量评估 2010年上海世博会是首次在中国举办的世界博览会。从1851年伦敦的“万国工业博览会”开始,世博会正日益成为各国人民交流历史文化、展示科技成果、体现合作精神、展望未来发展等的重要舞台。请你们选择感兴…...

使用nginx配置静态页面展示
文章目录 前言正文安装nginx配置 前言 目前有一系列html文件,比如sphinx通过make html输出的文件,需要通过ip远程访问,这就需要ngnix 主要内容参考:https://blog.csdn.net/qq_32460819/article/details/121131062 主要针对在do…...

[IOI2018] werewolf 狼人(Kruskal重构树 + 主席树)
https://www.luogu.com.cn/problem/P4899 首先,我们肯定要建两棵Kruskal重构树的,然后判两棵子树是否有相同编号节点 这是个经典问题,我们首先可以拍成dfs序,然后映射过去,然后相当于是判断一个区间是否有 [ l , r …...

snmpgetnext使用说明
1.snmpgetnext介绍 snmpgetnext命令是用来获取下一个节点的OID的值。 2.snmpgetnext安装 1.snmpgetnext安装 命令: yum -y install net-snmp net-snmp-utils [root@logstash ~]# yum -y install net-snmp net-snmp-utils Loaded plugins: fastestmirror Loading mirror …...

frameworks 之 触摸事件窗口查找
frameworks 之 触摸事件窗口查找 1. 初始化数据2. 查找窗口3. 分屏处理4. 检查对应的权限5.是否需要将事件传递给壁纸界面6. 成功处理 触摸流程中最重要的流程之一就是查找需要传递输入事件的窗口,并将触摸事件传递下去。 涉及到的类如下 frameworks/native/service…...

memset的用法
memset 是 C 语言标准库中的一个函数,用于将一块内存区域设置为特定的值。它的原型如下: c void *memset(void *s, int c, size_t n); - s 参数是要被填充的内存块的起始地址。 - c 参数是要填充的值。这个值会被转换为无符号字符,然后用来…...

阿里云国际站DDoS高防增值服务怎么样?
利用国外服务器建站的话,选择就具有多样性了,相较于我们常见的阿里云和腾讯云,国外的大厂商还有谷歌云,微软云,亚马逊云等,但是较之这些,同等产品进行比较的话,阿里云可以说当之无愧…...

open-cd中的changerformer网络结构分析
open-cd 目录 open-cd1.安装2.源码结构分析主干网络1.1 主干网络类2.neck2.Decoder3.测试模型6. changer主干网络 总结 该开源库基于: mmcv mmseg mmdet mmengine 1.安装 在安装过程中遇到的问题: 1.pytorch版本问题,open-cd采用的mmcv版本比…...

太速科技-426-基于XC7Z100+TMS320C6678的图像处理板卡
基于XC7Z100TMS320C6678的图像处理板卡 一、板卡概述 板卡基于独立的结构,实现ZYNQ XC7Z100DSP TMS320C6678的多路图像输入输出接口的综合图像处理,包含1路Camera link输入输出、1路HD-SDI输入输出、1路复合视频输入输出、2路光纤等视频接口,…...

asp.net Core 自定义中间件
内联中间件 中间件转移到类中 推荐中间件通过IApplicationBuilder 公开中间件 使用扩展方法 调用中间件 含有依赖项的 》》》中间件 参考资料...

掌握 C# 设计模式:从基础到依赖注入
设计模式是一种可以在开发中重复使用的解决方案,能够提高代码的可维护性、扩展性和复用性。C# 中常见的设计模式包括单例模式、工厂模式、观察者模式、策略模式等。本文将介绍这些常见的设计模式,并探讨 SOLID 原则和依赖注入(Dependency Inj…...

根据json转HttpClient脚本
String json “{\n” " “paths”: {\n" " “/dev-api/system/subjectResult/exportUserList”: {\n" " “post”: {\n" " “tags”: [\n" " “bd-subject-result-controller”\n" " ],\n" " “summ…...

如何将LiDAR坐标系下的3D点投影到相机2D图像上
将激光雷达点云投影到相机图像上做数据层的前融合,或者把激光雷达坐标系下标注的物体点云的3d bbox投影到相机图像上画出来,都需要做点云3D点坐标到图像像素坐标的转换计算,也就是LiDAR 3D坐标转像素坐标。 看了网上一些文章都存在有错误或者…...

JAVA就业笔记6——第二阶段(3)
课程须知 A类知识:工作和面试常用,代码必须要手敲,需要掌握。 B类知识:面试会问道,工作不常用,代码不需要手敲,理解能正确表达即可。 C类知识:工作和面试不常用,代码不…...

02.04、分割链表
02.04、[中等] 分割链表 1、题目描述 给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。 你不需要 保留 每个分区中各节点的初始相对位置。 2、解题思路 本题要求将链表分隔…...

Excel 中根据患者的就诊时间标记病例为“初诊”或“复诊”
1. 假设: 患者表:包含患者的基本信息,如患者 ID 和患者姓名。 病例表:包含病例信息,如患者 ID、就诊时间和就诊状态。 2. 操作步骤: 合并数据: 确保病例表中有一列包含患者 ID,以…...

遇到“mfc100u.dll丢失”的系统错误要怎么处理?科学修复mfc100u.dll
遇到“mfc100u.dll丢失”的系统错误会非常麻烦,因为mfc100u.dll是Microsoft Visual C 2010 Redistributable Package的重要部分,许多应用程序和游戏在运行时都需要调用这个文件。如果这个文件缺失,可能会导致相关软件或游戏启动失败。面对这种…...

[Linux] 逐层深入理解文件系统 (1)—— 进程操作文件
标题:[Linux] 文件系统 (1)—— 进程操作文件 个人主页水墨不写bug (图片来源于网络) 目录 一、进程与打开的文件 二、文件的系统调用与库函数的关系 1.系统调用open() 三、内存中的文件描述符表 四、缓冲区…...

RT-Thread 互斥量的概念
目录 概述 1 互斥量定义 1.1 概念介绍 1.2 线程优先级翻转问题 2 互斥量管理 2.1 结构体定义 2.2 函数接口介绍 2.2.1 rt_mutex_create函数 2.2.2 rt_mutex_delete 函数 2.2.3 初始化和脱离互斥量 概述 本文主要介绍互斥量的概念,实现原理。还介绍RT-Thre…...

6.计算机网络_UDP
UDP的主要特点: 无连接,发送数据之前不需要建立连接。不保证可靠交付。面向报文。应用层给UDP报文后,UDP并不会抽象为一个一个的字节,而是整个报文一起发送。没有拥塞控制。网络拥堵时,发送端并不会降低发送速率。可以…...

Windows应急响蓝安服面试
Windows应急响应 蓝队溯源流程 学习Windows应急首先要站在攻击者的角度去学习一些权限维持和权限提升的方法.,文章中的方法其实和内网攻防笔记有类似l红队教你怎么利用 蓝队教你怎么排查 攻防一体,应急响应排查这些项目就可以 端口/服务/进程/后门文件都是为了权限维持,得到s…...

PCL 点云配准-4PCS算法(粗配准)
目录 一、概述 1.1原理 1.2实现步骤 1.3应用场景 二、代码实现 2.1关键函数 2.1.1 加载点云数据 2.1.2 执行4PCS粗配准 2.1.3 可视化源点云、目标点云和配准结果 2.2完整代码 三、实现效果 3.1原始点云 3.2配准后点云 PCL点云算法汇总及实战案例汇总的目录地址链接…...

12、论文阅读:利用生成对抗网络实现无监督深度图像增强
Towards Unsupervised Deep Image Enhancement With Generative Adversarial Network 摘要介绍相关工作传统图像增强基于学习的图像增强 论文中提出的方法动机和目标网络架构损失函数1) 质量损失2) 保真损失3)身份损失4)Total Loss 实验 摘要 提高图像的…...