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坐标转像素坐标。 看了网上一些文章都存在有错误或者…...
Spring Cloud AWS 实战教程:构建高可用 SQS 消息队列应用 [特殊字符]
Spring Cloud AWS 实战教程:构建高可用 SQS 消息队列应用 🚀 【免费下载链接】spring-cloud-aws The New Home for Spring Cloud AWS 项目地址: https://gitcode.com/gh_mirrors/sp/spring-cloud-aws Spring Cloud AWS 是一个强大的开源框架&…...
三步实现跨架构程序兼容:Box64高效架构转换指南
三步实现跨架构程序兼容:Box64高效架构转换指南 【免费下载链接】box64 Box64 - Linux Userspace x86_64 Emulator with a twist, targeted at ARM64, RV64 and LoongArch Linux devices 项目地址: https://gitcode.com/gh_mirrors/bo/box64 你是否曾在ARM64…...
全链路压测实战:双十一级别的流量,我是这样扛住的
作为一名在质量保障领域摸爬滚打多年的测试工程师,我深知传统的单接口压测在如今分布式架构下的无力感。当业务流量达到双十一这种脉冲式、高并发的级别时,任何一个非核心链路上的“短板”都可能引发系统性的雪崩。全链路压测不再是选择题,而…...
新能源车轻量化为什么开始盯上高强镁合金?
续航,是悬在每一台纯电动汽车头上的达摩克利斯之剑。多充一度电、多堆一些正极材料,是一条路;但还有另一条路——把车造得更轻。 SAE(美国汽车工程师学会)的测算已经被反复引用:整车每减重100千克ÿ…...
基于MAX78000与CNN的智能螺栓巡检小车:嵌入式AI实战解析
1. 项目概述与核心思路在轨道交通的日常运维中,螺栓的紧固状态检查是一项繁重且关键的任务。无论是轨道上的紧固螺栓,还是列车转向架、轮对轴承上的关键螺栓,其松动或失效都可能引发严重的安全事故。传统的人工巡检方式不仅效率低下ÿ…...
从《吃豆人》到开放世界:聊聊Unity Navigation里Agent Radius和Cost的那些‘潜规则’
从《吃豆人》到开放世界:Unity Navigation中Agent Radius与Cost的隐藏逻辑1980年诞生的《吃豆人》用简单的迷宫路径定义了早期游戏AI的移动规则——幽灵们沿着固定路线巡逻,遇到转角时随机选择方向。这种设计在当时堪称革命性,但以今天的标准…...
Fiddler手机断网真相:TLS握手与证书固定的协议级拦截
1. 为什么Fiddler一开,手机就断网?这不是配置问题,是协议层的“信任危机”Fiddler抓包手机流量,本该是移动开发、测试、安全分析中最基础的操作之一。但几乎每个刚上手的人,都会在第二天早上发现:手机Wi-Fi…...
鸿蒙HarmonyOS 5与Unity跨运行时通信实战指南
1. 这不是“调个API”那么简单:为什么鸿蒙Unity通信总在临门一脚卡住我第一次把Unity打包的AR模块塞进HarmonyOS 5 App里时,信心满满——毕竟文档里写着“支持JS/ArkTS调用Native能力”,Unity也标榜“跨平台通用”。结果呢?App一启…...
“--glow”并不存在?!深度逆向Midjourney 6.1源码级辉光模拟协议,曝光官方刻意隐藏的4个隐式辉光增强开关
更多请点击: https://kaifayun.com 第一章:辉光效果的视觉本质与Midjourney 6.1协议悖论 辉光(Glow)并非物理光源的直接投射,而是人眼视网膜对高对比度边缘与饱和色域交界处产生的神经光学响应——一种由局部亮度梯度…...
DeepSeek注释质量跃迁路径(附12个真实项目对比数据+可复用Prompt模板)
更多请点击: https://codechina.net 第一章:DeepSeek注释质量跃迁路径(附12个真实项目对比数据可复用Prompt模板) 高质量代码注释不再是“锦上添花”,而是模型理解意图、团队高效协同与长期可维护性的核心基础设施。…...
