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坐标转像素坐标。 看了网上一些文章都存在有错误或者…...
AtCoder 第409场初级竞赛 A~E题解
A Conflict 【题目链接】 原题链接:A - Conflict 【考点】 枚举 【题目大意】 找到是否有两人都想要的物品。 【解析】 遍历两端字符串,只有在同时为 o 时输出 Yes 并结束程序,否则输出 No。 【难度】 GESP三级 【代码参考】 #i…...

前端导出带有合并单元格的列表
// 导出async function exportExcel(fileName "共识调整.xlsx") {// 所有数据const exportData await getAllMainData();// 表头内容let fitstTitleList [];const secondTitleList [];allColumns.value.forEach(column > {if (!column.children) {fitstTitleL…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...

Unsafe Fileupload篇补充-木马的详细教程与木马分享(中国蚁剑方式)
在之前的皮卡丘靶场第九期Unsafe Fileupload篇中我们学习了木马的原理并且学了一个简单的木马文件 本期内容是为了更好的为大家解释木马(服务器方面的)的原理,连接,以及各种木马及连接工具的分享 文件木马:https://w…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...

PHP 8.5 即将发布:管道操作符、强力调试
前不久,PHP宣布了即将在 2025 年 11 月 20 日 正式发布的 PHP 8.5!作为 PHP 语言的又一次重要迭代,PHP 8.5 承诺带来一系列旨在提升代码可读性、健壮性以及开发者效率的改进。而更令人兴奋的是,借助强大的本地开发环境 ServBay&am…...

Ubuntu Cursor升级成v1.0
0. 当前版本低 使用当前 Cursor v0.50时 GitHub Copilot Chat 打不开,快捷键也不好用,当看到 Cursor 升级后,还是蛮高兴的 1. 下载 Cursor 下载地址:https://www.cursor.com/cn/downloads 点击下载 Linux (x64) ,…...
在 Spring Boot 项目里,MYSQL中json类型字段使用
前言: 因为程序特殊需求导致,需要mysql数据库存储json类型数据,因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...

【UE5 C++】通过文件对话框获取选择文件的路径
目录 效果 步骤 源码 效果 步骤 1. 在“xxx.Build.cs”中添加需要使用的模块 ,这里主要使用“DesktopPlatform”模块 2. 添加后闭UE编辑器,右键点击 .uproject 文件,选择 "Generate Visual Studio project files",重…...

第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10+pip3.10)
第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10pip3.10) 一:前言二:安装编译依赖二:安装Python3.10三:安装PIP3.10四:安装Paddlepaddle基础框架4.1…...