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坐标转像素坐标。 看了网上一些文章都存在有错误或者…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...
pikachu靶场通关笔记22-1 SQL注入05-1-insert注入(报错法)
目录 一、SQL注入 二、insert注入 三、报错型注入 四、updatexml函数 五、源码审计 六、insert渗透实战 1、渗透准备 2、获取数据库名database 3、获取表名table 4、获取列名column 5、获取字段 本系列为通过《pikachu靶场通关笔记》的SQL注入关卡(共10关࿰…...
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
以下是一个完整的 Angular 微前端示例,其中使用的是 Module Federation 和 npx-build-plus 实现了主应用(Shell)与子应用(Remote)的集成。 🛠️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...
在QWebEngineView上实现鼠标、触摸等事件捕获的解决方案
这个问题我看其他博主也写了,要么要会员、要么写的乱七八糟。这里我整理一下,把问题说清楚并且给出代码,拿去用就行,照着葫芦画瓢。 问题 在继承QWebEngineView后,重写mousePressEvent或event函数无法捕获鼠标按下事…...
R语言速释制剂QBD解决方案之三
本文是《Quality by Design for ANDAs: An Example for Immediate-Release Dosage Forms》第一个处方的R语言解决方案。 第一个处方研究评估原料药粒径分布、MCC/Lactose比例、崩解剂用量对制剂CQAs的影响。 第二处方研究用于理解颗粒外加硬脂酸镁和滑石粉对片剂质量和可生产…...
安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲
文章目录 前言第一部分:体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分:体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...
JavaScript基础-API 和 Web API
在学习JavaScript的过程中,理解API(应用程序接口)和Web API的概念及其应用是非常重要的。这些工具极大地扩展了JavaScript的功能,使得开发者能够创建出功能丰富、交互性强的Web应用程序。本文将深入探讨JavaScript中的API与Web AP…...
宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...
协议转换利器,profinet转ethercat网关的两大派系,各有千秋
随着工业以太网的发展,其高效、便捷、协议开放、易于冗余等诸多优点,被越来越多的工业现场所采用。西门子SIMATIC S7-1200/1500系列PLC集成有Profinet接口,具有实时性、开放性,使用TCP/IP和IT标准,符合基于工业以太网的…...
Visual Studio Code 扩展
Visual Studio Code 扩展 change-case 大小写转换EmmyLua for VSCode 调试插件Bookmarks 书签 change-case 大小写转换 https://marketplace.visualstudio.com/items?itemNamewmaurer.change-case 选中单词后,命令 changeCase.commands 可预览转换效果 EmmyLua…...
