0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Java的模块系统定义

科技绿洲 来源:Java技术指北 作者:Java技术指北 2023-10-11 16:01 次阅读

前言

对于模块的定义,相信小伙伴们都不会有太大的疑问,这里引用JDK中模块的定义:一组可重用的相关包和资源,以及模块的描述信息。直白点的描述就是:用一种比package更大级别的组织方法来管理我们的类。讲到它的作用是组织管理,大家是不是就开始联想到了OSGi,JBoss Module,Maven,甚至是微服务呢?首先,这几种形式都是用于软件的模块化方法,只是应用的场景和各自的长处有所不同,指北君用一张图表来做一下简单对比:

图片
模块化方式

  • 微服务:服务粒度的组织方法,比模块级别更高
  • Maven:依赖管理工具

变化

首先,作为一头猿,最直接的编码IDE工具,以Eclipse为例,我们新建一个标准的Java工程,切换下JDK进行对比,

图片
JDK库-JAVA11

图片
JDK库-JAVA8

首先我们会发现:项目使用中的JDK11的库中比之前JDK8多出很多条目来,而且和之前的完全不一样了。这就是JDK引入模块后,将原来的库进行了拆分。为什么要拆呢?

  1. 伴随Java的版本升级,包越来越多,功能越来越多,组织管理越来越难
  2. 一些jar包太大(比如rt.jar),不利于在小型设备中运行
  3. 无法定义只能在模块内部访问的接口,只能通过一些额外的约束,比如文档,internal等形式进行提示

JDK是如何对原有的jar进行拆分的呢?我们查看模块名称会发现,所有模块分成两类:java开头和jdk开头,这是按照JAVA的JRE和JDK范围进行的大类别的划分,然后再按照功能级做进一步划分。除了在开发者环境中引入的库发生变化外,在JDK的安装目录中也有类似的变化

图片
JDK安装目录对比

明显是jre目录不在了,增加了jmods目录,lib下面的组织形式也发生了较大的变化。

可以做什么

前面我们介绍模块系统引入后带来的直观的变化,这一节指北君要介绍模块系统的作用,我们先来看一下模块的定义里面包含的要素:

  • 名称 – 模块名
  • 依赖 - 本模块依赖的一系列其他模块
  • 公共包 - 外部可访问的所有包
  • 提供的服务 - 提供其他模块消费的服务实现
  • 消费的服务 - 允许当前模块作为服务消费者
  • 反射权限 - 显式允许其他类通过反射访问的包的私有成员

从定义的要素中我们发现:模块不仅仅是提供的一直包、类的组织方式,更重要的是提供了以前无法支持的安全访问控制。

模块的类型

有四种类型的模块

  • 系统模块:JDK定义的模块,可以通过下面命令来获取
java --list-modules
  • 应用模块:我们开发的应用对应的模块,通过module-info.java定义并编译成对应的class
  • 自动模块:通过模块路径加载的第三方jar包
  • 未命名模块:不是通过模块路径加载的第三方jar

模块的声明

要创建一个模块,需要在包的根路径下创建module-info.java(注意名称是固定的),如果按照Class的方式创建会出现名称校验失败,这时候可以直接创建一个文件命名为module-info.java。

module moduleName {
    
}

使用关键字module定义,模块名称按照约定为通过点号"."分割的小写词组,比如java.base, north.sample。

requires

requires用来管理模块的依赖关系,我们一旦采用了模块,我们会发现原来的有些类会出现编译错误,这是因为我们的代码中应用了默认之外的包,需要通过requires关键词引入我们引用的模块。

module north.smaple{
    requires java.scripting;
    
}

使用requires还可以使用两个修饰词:static,transitive,

  • static用来定义一个可选的模块依赖,仅当编译时有依赖才引用
  • transitive 表示下游的模块不用显式声明,就可以使用上级模块中通过transitive关键字引入的模块

exports

通过exports我们可以定义可访问的接口

module north.smaple{ 
    exports north.jdk.scripting;
}

我们还可以通过exports…to来定义接口开放的目标对象。

uses

定义使用的服务,以java.sql模块代码为例:

module java.sql {
    ...

    exports java.sql;
    exports javax.sql;

    uses java.sql.Driver;
}

为什么对服务特殊处理呢?各位小伙伴是不是觉得:requires就能够定义访问依赖,为啥还要用uses呢?这是因为,uses相对于requires是存在区别的。比如,我们的服务接口实现和服务接口不在同一个模块中,如果用requires则需要对所有的实现模块引入,如果使用uses只需要引入接口所在的模块就行了,是不是很方便呢!而且有时我们都不知道接口的实现模块,这时候都无法通过requires引入。

provide

如果需要定义外部可使用的服务,则通过provide声明,语法是 provides <服务接口> with <服务实现>

module north.smaple{
    provides ISampleService with ISampleServiceImpl;
}

open/opens

open是用来定义模块的被外部模块通过反射调用的权限,在这之前我们可以通过反射调用任何我们想要访问的类型和成员,甚至是私有属性的。在使用模块系统后,如果我们要保持之前的全访问,可以直接在module前添加open关键字。

open module north.smaple{
    ...
}

如果我们有针对性的开放反射权限,则通过opens

module north.smaple{
    opens north.jdk.scripting;
}

opens还支持更严格的定义 opens ... to ...

总结

关于Java的模块系统,我们今天就学习到这里,相信经过代码示例和讲解,各位小伙伴已经可以将模块系统应用到项目中了。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 模块
    +关注

    关注

    7

    文章

    2484

    浏览量

    46533
  • JAVA
    +关注

    关注

    19

    文章

    2904

    浏览量

    102995
  • 软件
    +关注

    关注

    67

    文章

    4350

    浏览量

    85624
  • 编码
    +关注

    关注

    6

    文章

    835

    浏览量

    54457
收藏 人收藏

    评论

    相关推荐

    Java Security的核心模块和代码演示

    ,包括其核心概念、关键模块以及具体应用。通过详细分析,希望帮助读者更好地理解如何在Java应用程序中实现安全防护,提高系统的可靠性和稳定性。 主要功能包括授权、访问控制、数据加密、身份验证等 核心
    的头像 发表于 10-07 15:44 794次阅读

    基于JAVA与SSM的移动电源租赁系统的设计

    涉及到的关键设计思想及重要作业流程作了具体分析和介绍,并对各个模块的设计思想及设计过程作了详细阐述。本系统的设计采用了JAVA技术,SSM框架,三层架构,以Oracle作为数据库支撑平台。该
    发表于 01-03 07:08

    Java语言基础

    Java语言基础4.1 Java 程序的构成4.2 数据类型、变量与常量4.3 表达式4.4 流程控制语句Java源程序是由类组成的,每个程序中可以定义若干个类,但只能有一个主类。在
    发表于 12-14 20:30 187次下载

    Java的类

    1.1 有关面向对象的概念1.2 Java语言的特点1.3 Java的开发环境1.4  第一个Java程序1.5 Java类型系统
    发表于 04-28 14:23 0次下载

    JAVA制作的ICQ系统

    JAVA制作的ICQ系统   基于JAVA的ICQ系统服务器与客户间通过套接口Socket(TCP)连接
    发表于 03-29 14:12 21次下载

    模块Java:动态模块

    在前一篇文章《模块Java:静态模块化》中,我们讨论了如何构建Java模 块并将其作为一个单独的JAR进行部署。文中的例子给出了一个client和一个 server bundle(两
    发表于 12-01 11:54 24次下载

    JAVA常用系统类的使用 实验

    实验 6 常用系统类的使用 一、实验目的 了解 Java 常用的系统类,包括 Java Applet、字符串类、输入输出流类、数学函数类、日期类、随机数类以及向量类等的基
    发表于 09-23 19:01 1690次阅读

    JAVA教程之自定义光标

    JAVA教程之自定义光标,很好的学习资料。
    发表于 03-31 11:13 7次下载

    新的Java 9功能介绍

    第9版给Java定义模块的可能性。一个模块是一个命名的,自描述的程序组件,由一个或多个包(和数据)组成。模块可以
    发表于 09-27 11:36 0次下载
    新的<b class='flag-5'>Java</b> 9功能介绍

    Java 9的新特性详解

    JDK 9的Java平台模块系统 Java 9最宝贵的特性,即模块化以Java平台
    发表于 09-28 14:37 0次下载

    Java 9的新特性总结

    模块系统Jigsaw 项目 模块化是一个很通用的概念。在软件中,模块化可以运用到编写和实现一个程序和计算系统,他们都是作为独立的
    发表于 09-28 20:04 0次下载
    <b class='flag-5'>Java</b> 9的新特性总结

    java中数组的三种定义方式_java中数组的定义及使用方法(推荐)

    java中,数组是一种很常用的工具,本文将介绍来java中数组的三种定义方式以及java中数组的定义及使用方法。
    发表于 01-29 09:53 3.2w次阅读

    Java定义常量两种常见策略

    关于Java中常量的话题似乎有很多困惑。有些人使用整数或字符串来定义常量,而另一些人则使用枚举。
    的头像 发表于 05-05 20:49 2136次阅读
    在<b class='flag-5'>Java</b>中<b class='flag-5'>定义</b>常量两种常见策略

    java源程序是由类定义组成的吗

    Java源程序是由类定义组成的。在Java中,类是用来定义对象的一种结构化方式。一个类可以包含字段(变量)和方法(函数),它们一起描述了对象的属性和行为。 在
    的头像 发表于 11-28 16:36 312次阅读

    java中长整型怎么定义

    Java中,长整型是一种数据类型,用于存储整数值。它的取值范围比整型要大,可以存储更大的整数值。长整型的大小为8字节,即64位。在Java中,长整型用关键字"long"进行声明和定义。 在J
    的头像 发表于 11-30 11:29 1792次阅读