【Java 基础篇】Java 模块化详解

在这里插入图片描述

Java 9引入了一项重要的功能:模块化(Module System)。模块化是一种将代码和资源封装到可重用和独立的单元中的方法,它有助于改善代码的可维护性、可重用性和安全性。本文将介绍Java模块化的基本概念、如何创建和使用模块以及一些最佳实践。

什么是Java模块化?

在Java 9之前,Java应用程序是以JAR文件的形式组织的,其中包含了一堆类和资源。这种方式存在一些问题:

  • 可维护性差:JAR文件可以包含大量的类和资源,这使得应用程序的结构变得混乱,难以维护。
  • 可重用性差:在多个应用程序之间共享代码和资源比较困难。
  • 安全性问题:所有的类都在同一个类路径中,这可能导致意外的访问和依赖关系。

Java模块化解决了这些问题。模块是一种新的编程单元,它可以包含类、资源和其他模块的依赖关系。模块化的代码更容易维护,更容易重用,同时也提供了更好的安全性。

模块化的基本概念

在开始使用Java模块化之前,让我们先了解一些基本概念:

1. 模块(Module)

一个模块是一个可重用的单元,它包含了一组相关的类和资源。每个模块都有一个名字,并可以声明自己的依赖关系。

2. 模块声明(Module Declaration)

一个模块声明是一个包含在module-info.java文件中的文件,它定义了一个模块的名称、依赖关系和其他特性。

3. 模块路径(Module Path)

模块路径是一组目录和JAR文件,其中包含了模块的JMOD文件和module-info.class文件。模块路径用于告诉JVM哪些模块可用。

4. 模块化 JAR 文件(Modular JAR File)

模块化JAR文件是一种特殊类型的JAR文件,它包含了一个模块的类和资源,以及module-info.class文件。

5. 自动模块(Automatic Module)

如果一个JAR文件没有module-info.class文件,它被称为自动模块。自动模块的名称基于JAR文件的文件名,并且具有一些默认的依赖关系。

6. 依赖性(Dependency)

一个模块可以声明对其他模块的依赖关系,以便在编译时和运行时使用其他模块的类和资源。

创建一个简单的模块

让我们从创建一个简单的Java模块开始,以便更好地理解模块化的概念。假设我们有一个应用程序,它有两个模块:一个模块用于处理数据库连接,另一个模块用于处理用户界面。

第一步:创建数据库模块

首先,我们创建一个数据库模块,它包含了一个简单的类DatabaseConnection

// DatabaseConnection.java
package com.example.database;public class DatabaseConnection {public void connect() {System.out.println("Connected to the database.");}
}

然后,我们为该模块创建一个module-info.java文件,声明该模块的名称:

// module-info.java
module databaseModule {
}

这个模块不依赖于其他模块,所以module-info.java文件中没有声明任何依赖关系。

第二步:创建用户界面模块

接下来,我们创建一个用户界面模块,它包含了一个简单的类UserInterface

// UserInterface.java
package com.example.ui;public class UserInterface {public void display() {System.out.println("User interface displayed.");}
}

为该模块创建一个module-info.java文件,声明该模块的名称和对数据库模块的依赖:

// module-info.java
module uiModule {requires databaseModule;
}

这个模块声明了对databaseModule的依赖,这意味着它可以使用databaseModule中的类和资源。

第三步:编译和运行模块

现在,我们可以使用Java 9编译器来编译这两个模块:

javac -d out/database databaseModule/*.java
javac -d out/ui --module-path out databaseModule/*.java uiModule/*.java

然后,我们可以运行UserInterface类来启动我们的应用程序:

java --module-path out -m uiModule/com.example.ui.UserInterface

这将会输出:

Connected to the database.
User interface displayed.

模块化的更多操作

当您在Java应用程序中使用模块化时,可以执行许多不同的操作,以更好地组织、管理和优化您的代码。以下是一些模块化的更多操作:

  1. 导出和打包模块:您可以使用exports关键字在module-info.java中声明哪些包可以被其他模块访问。这允许您控制哪些部分的代码对外可见。例如:

    module myModule {exports com.example.mypackage;
    }
    

    您还可以使用opens关键字导出包,以便其他模块可以反射地访问包中的非公开类型。

  2. 开放模块:如果您希望模块对所有其他模块开放,可以使用open关键字。这对于编写插件或扩展模块很有用。

    module myModule {open com.example.mypackage;
    }
    
  3. 命令行工具:Java 9引入了一些命令行工具,如jdeps,用于分析模块之间的依赖关系,以及jlink,用于创建自定义运行时映像。

  4. 运行时图像:您可以使用jlink命令将您的模块化应用程序与JRE一起打包成自定义运行时图像。这有助于减小应用程序的大小,因为只包括了应用程序所需的模块。

  5. 模块路径:模块路径是一个包含模块的目录或JAR文件的集合,它用于在运行时加载模块。您可以使用--module-path选项来指定模块路径。

  6. 自动模块:如果您的应用程序包含非模块化的JAR文件,Java会自动将它们转换为自动模块,以便它们可以与模块一起使用。

  7. 模块化JAR文件:您可以使用jar工具创建模块化的JAR文件,其中包含了模块描述文件(module-info.class)。

  8. 模块化库:许多常用的Java库已经进行了模块化,以便更好地与模块化应用程序集成。您可以在模块路径上指定这些库,而无需手动管理它们的依赖关系。

  9. 模块化测试:使用模块路径和--module选项,您可以在单元测试中模拟模块化环境。

  10. 版本管理:在module-info.java中可以使用requires static关键字来声明可选依赖关系,这些依赖关系只在模块可用时才会生效。

这些是模块化Java应用程序中的一些更多操作。模块化使得Java应用程序更易于维护和扩展,同时提供了更好的封装和可重用性。根据您的项目需求,您可以选择适当的操作来更好地利用模块化的优势。

模块化的最佳实践

以下是一些模块化的最佳实践:

  1. 模块命名规范:给模块取一个有意义的名字,通常使用逆域名表示法(例如:com.example.myapp)。

  2. 明确的依赖关系:在module-info.java文件中明确声明模块的依赖关系,以确保应用程序的模块之间的依赖关系清晰可见。

  3. 最小依赖原则:尽量减少模块之间的依赖关系,只依赖于真正需要的模块。

  4. 版本化的依赖关系:如果可能的话,使用版本化的依赖关系来确保模块依赖的是正确的版本。

  5. 单一责任原则:将每个模块限制为一个特定的功能或领域,以提高可维护性和可重用性。

  6. 测试和验证:确保模块之间的依赖关系和交互在编译时和运行时都能正常工作。

  7. 模块路径管理:管理模块路径以确保应用程序能够正确加载和运行。

注意事项

在编写和使用模块化的Java应用程序时,有一些重要的注意事项,以确保您的应用程序正确运行和维护。以下是一些模块化的注意事项:

  1. 模块依赖关系:仔细考虑您的模块之间的依赖关系。确保模块之间的依赖关系是明确的,避免循环依赖。使用requires语句声明依赖关系,并根据需要使用requires transitiverequires static

  2. 版本管理:了解模块之间的版本管理。Java 9引入了模块化版本的概念,允许模块依赖于特定版本的其他模块。考虑使用requires static来声明可选的、仅在特定版本下才有效的依赖关系。

  3. 模块命名:为您的模块选择合适的名称。模块名称应该唯一且易于理解。遵循Java的包命名约定,使用反向域名(例如com.example.mymodule)。

  4. 模块路径:在运行应用程序时,使用--module-path选项指定模块路径。确保正确设置模块路径,以便Java可以找到并加载您的模块。

  5. 非模块化库:如果您使用了非模块化的JAR文件,将其包装为自动模块或创建模块化的版本。非模块化库的依赖关系可能会引入复杂性。

  6. 模块化库:考虑使用已经模块化的库,以减少与模块路径和版本管理相关的问题。

  7. 运行时图像:如果您使用jlink创建自定义运行时图像,请确保包括了所有必要的模块,并排除不必要的模块,以减小应用程序的大小。

  8. 测试:编写单元测试以确保模块化应用程序的正确性。使用模块路径和--module选项来模拟模块化环境进行测试。

  9. 模块描述文件:模块描述文件(module-info.java)是模块化应用程序的关键组成部分。确保正确声明依赖关系、导出和打包模块,以及使用其他关键字来管理可见性。

  10. 模块间通信:模块之间的通信应该在依赖模块的基础上进行。不要尝试绕过模块系统的可见性控制。

  11. 跨模块访问:如果需要在模块之间共享数据或访问非公开成员,请使用opensopens...to语句,以允许受信任的模块进行反射操作。

  12. 性能和内存开销:模块化应用程序的启动时间和内存开销可能会有所增加。在部署和测试应用程序时,要考虑性能方面的因素。

  13. 迁移:如果您正在迁移现有的应用程序到模块化架构,确保逐步迁移,以减少中断和问题。

  14. 文档和培训:为开发团队提供关于模块化的文档和培训,以确保所有开发人员都理解和遵守模块化的最佳实践。

  15. 工具支持:使用Java 9及更高版本,以充分利用模块化系统和相关的工具,如jdepsjlinkjmod

这些注意事项有助于确保您的模块化Java应用程序能够正确运行

结论

Java模块化是一个强大的工具,可以帮助您更好地组织和管理代码,提高可维护性和可重用性。本文介绍了模块化的基本概念,以及如何创建和使用模块。遵循最佳实践,可以使您的Java应用程序更加健壮和可维护。希望本文对您理解Java模块化有所帮助。

作者信息

作者 : 繁依Fanyi
CSDN: https://techfanyi.blog.csdn.net
掘金:https://juejin.cn/user/4154386571867191

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/87587.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI数字人:最强声音驱动面部表情模型VideoReTalking

目录 1 VideoReTalking论文解读 1.1 介绍 1.2 相关工作 1.2.1 视频编辑中的音频配音 1.2.2 基于音频的单图像面部动画 1.3 框架 1.3.1 语义引导重演网络 1.3.2 口型同步网络 1.3.3 身份感知增强网络 1.3.4 后期处理 1.4 训练 1.4.1 每个模块的训练 1.4.2 评估 1.…

JSP ssm 零配件管理系统myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 java ssm 零配件管理系统是一套完善的web设计系统(系统采用SSM框架进行设计开发,springspringMVCmybatis),对理解JSP java编程开发语言有帮助,系统具有完整的源代码和数据库,系统主要采用…

注入之SQLMAP(工具注入)

i sqlmap是一个自动化的SQL注入工具,其主要功能是扫描,发现并利用给定的URL和SQL注入漏洞,其广泛的功能和选项包括数据库指纹,枚举,数据库提权,访问目标文件系统,并在获取操作权限时执行任…

RabbitMQ生产故障问题分析

1. 问题引发 由某个服务BI-collector-xx队列出现阻塞,影响很整个rabbitMQ集群服务不可用,多个应用MQ生产者服务出现假死状态,系统影响面较广,业务影响很大。当时为了应急处理,恢复系统可用,运维相对粗暴的把…

java Spring Boot2.7实现一个简单的爬虫功能

首先 我们要在 pom.xml 中注入Jsoup 这是一个简单的java爬虫框架 <dependency><groupId>org.jsoup</groupId><artifactId>jsoup</artifactId><version>1.14.1</version> </dependency>然后这里我们直接用main吧 做简单一点 我…

分布式搜索引擎01

1.初识elasticsearch 1.1.了解ES 1.1.1.elasticsearch的作用 elasticsearch是一款非常强大的开源搜索引擎,具备非常多强大功能,可以帮助我们从海量数据中快速找到需要的内容 例如: 在GitHub搜索代码 在电商网站搜索商品 在百度搜索答案 在打车软件搜索附近的车 1.1.2.ELK…

Java函数式接口(Consumer、Function、Predicate、Supplier)详解及代码示例

函数式接口 java.util.function : Consumer :消费型函数接口 void accept(T t) Function :函数型接口 R apply(T t) Predicate :判断型接口 boolean test(T t) Supplier :供给型接口 T get() Consumer - 消费型函数接口 该接口代表了一个接受一个参数并且不返回结果的操作。…

【Less-CSS】初识Less,使编写 CSS 变得简洁

初识Less&#xff0c;使编写 CSS 变得简洁 1.Less简述2.LESS 原理及使用方式3.示例4.less语法5.Easy Less插件 作为一门标记性语言&#xff0c;CSS 的语法相对简单&#xff0c;对使用者的要求较低&#xff0c;但同时也带来一些问题&#xff1a;CSS 需要书写大量看似没有逻辑的代…

Python爬虫从端到端抓取网页

网页抓取和 REST API 简介 网页抓取是使用计算机程序以自动方式从网站提取和解析数据的过程。这是创建用于研究和学习的数据集的有用技术。虽然网页抓取通常涉及解析和处理 HTML 文档&#xff0c;但某些平台还提供 REST API 来以机器可读格式&#xff08;如 JSON&#xff09;检…

【C++】C++ 类中的 this 指针用法 ③ ( 全局函数 与 成员函数 相互转化 | 有参构造函数设置默认参数值 | 返回匿名对象与返回引用 )

文章目录 一、全局函数 与 成员函数 相互转化1、成员函数转为全局函数 - 多了一个参数2、全局函数转为成员函数 - 通过 this 指针隐藏操作数 二、有参构造函数设置默认参数值三、返回匿名对象与返回引用四、完整代码示例 一、全局函数 与 成员函数 相互转化 1、成员函数转为全局…

一、vue2的基础语法巩固

一、定义&#xff1a;是一个渐进式的JavaScript框架 二、特点&#xff1a; 减少了大量的DOM操作编写 &#xff0c;可以更专注于逻辑操作分离数据和界面的呈现&#xff0c;降低了代码耦合度(前端端分离)支持组件化开发&#xff0c;更利于中大型项目的代码组织 vue2核心功能&a…

【Linux】生产消费模型 + 线程池

文章目录 &#x1f4d6; 前言1. 生产消费模型2. 阻塞队列2.1 成员变量&#xff1a;2.2 入队(push)和出队(pop)&#xff1a;2.3 封装与测试运行&#xff1a;2.3 - 1 对代码进一步封装2.3 - 2 分配运算任务2.3 - 3 测试与运行 3. 循环阻塞队列3.1 POSIX信号量&#xff1a;3.1 - 1…

WKB近似

WKB方法用于研究一种特定类型的微分方程的全局性质 很有用这种特定的微分方程形如&#xff1a; 经过一些不是特别复杂的推导&#xff0c;我们可以得到他的WKB近似解。 该近似解的选择取决于函数和参数的性质同时&#xff0c;我们默认函数的定义域为当恒大于零,时&#xff1a; 当…

44.java教程

目录 一、Java 教程。 &#xff08;1&#xff09;我的第一个 JAVA 程序。 &#xff08;2&#xff09;Java 简介。 &#xff08;2.1&#xff09;java简介。 &#xff08;2.2&#xff09;主要特性。 &#xff08;2.3&#xff09;发展历史。 &#xff08;2.4&#xff09;J…

iOS应用程序的签名、重签名和安装测试

目录 前言 打开要处理的IPA文件 设置签名使用的证书和描述文件 开始ios ipa重签名 前言 ipa编译出来后&#xff0c;或者ipa进行修改后&#xff0c;需要进行重新签名才能安装到测试手机&#xff0c;或者提交app store供apple 商店审核上架。ipaguard有签名和重签名功能&…

吴恩达ChatGPT《Finetuning Large Language Models》笔记

课程地址&#xff1a;https://learn.deeplearning.ai/finetuning-large-language-models/lesson/1/introduction Introduction 动机&#xff1a;虽然编写提示词&#xff08;Prompt&#xff09;可以让LLM按照指示执行任务&#xff0c;比如提取文本中的关键词&#xff0c;或者对…

React中setState的原理及深层理解

1.为什么使用setState React并没有实现类似于Vue2中的Object.defineProperty或者Vue3中的Proxy的方式来监听数据的变化 我们必须通过setState来告知React数据已经发生了变化 setState方法是从Component中继承过来的。 2.setState异步更新 setState设计为异步&#xff0c;可…

PHY6230低成本遥控灯控芯片国产蓝牙BLE5.2 2.4G SoC

高性价比的低功耗高性能蓝牙5.2系统级芯片&#xff0c;适用多种PC/手机外设连接场景。 高性能多模射频收发机&#xff1a; 通过硬件模块的充分复用实现高性能多模数字收发机。发射机&#xff0c;最大发射功率10dBm&#xff1b;BLE 1Mbps速率接收机灵敏度达到-96dBm&#xff1…

解决Vue设置图片的动态src不生效的问题

一、问题描述 在vue项目中&#xff0c;想要动态设置img的src时&#xff0c;此时发现图片会加载失败。在Vue代码中是这样写的&#xff1a; 在Vue的data中是这样写的&#xff1a; 我的图片在根目录下的static里面&#xff1a; 但是在页面上这个图片却无法加载出来。 二、解决方案…

五、核支持向量机算法(NuSVC,Nu-Support Vector Classification)(有监督学习)

和支持向量分类(Nu-Support Vector Classification)&#xff0c;与 SVC 类似&#xff0c;但使用一个参数来控制支持向量的数量&#xff0c;其实现基于libsvm 一、算法思路 本质都是SVM中的一种优化&#xff0c;原理都类似&#xff0c;详细算法思路可以参考博文&#xff1a;三…