Java模块系统从Java 9引入,通过module-info.java定义模块化结构,实现强封装、可靠配置和精简运行时。模块使用module声明身份,requires指定依赖,exports控制包导出,opens支持反射访问,uses与provides实现服务加载机制,从而替代传统类路径的模糊性,提升大型项目可维护性。

Java模块系统(Module System)是从Java 9开始引入的一项重要特性,目的是解决大型项目中类路径(classpath)的混乱问题,提升程序的封装性与可维护性。理解Java模块机制,关键在于掌握“模块是什么”、“为什么需要它”以及“如何使用它”。
什么是Java模块
一个Java模块是一个命名的、自描述的代码单元。它通过一个名为 module-info.java 的源文件来声明自己的身份和依赖关系。这个文件位于模块的根目录下,内容使用 module 关键字定义。
例如:
module com.example.mymodule {requires java.base;
requires com.example.service;
exports com.example.api;
}
上面这段代码定义了一个叫 com.example.mymodule 的模块,它依赖于 java.base 和另一个自定义模块 com.example.service,同时它只对外公开 com.example.api 包中的类。
立即学习“Java免费学习笔记(深入)”;
模块的核心作用
模块机制主要解决了以下几个长期困扰Java开发者的问题:
- 强封装:在模块之前,即使某个类没有被设计为公开使用,只要它在classpath上,其他代码就可以访问。模块允许你明确指定哪些包是导出的(exports),未导出的包默认不可见。
- 可靠的配置:JVM在启动时会检查模块之间的依赖是否完整且无冲突,避免运行时才发现缺少类。
- 精简运行时:借助jlink工具,可以根据模块依赖生成定制的JRE,减少部署体积。
- 替代模糊的类路径机制:模块路径(module path)取代了传统类路径的部分功能,使得依赖更清晰。
模块的关键语法和概念
理解以下关键字对掌握模块机制至关重要:
- module 模块名:定义当前模块的身份。
- requires 模块名:声明本模块依赖的其他模块。比如 requires java.sql 表示依赖Java的SQL模块。
- exports 包名:声明该包内的公共类可以被其他模块访问。
- opens 包名:允许反射访问该包中的类(如用于JPA、Jackson等框架),但不开放普通访问。
- uses 接口名:声明本模块使用某个服务接口,配合SPI机制。
- provides ... with ...:实现服务提供者机制,表示本模块提供了某个接口的具体实现。
实际使用场景示例
假设你开发一个Web服务模块,结构如下:
src/└── com.myweb.module/
└── module-info.java
└── com/
└── myweb/
├── controller/
├── service/
└── internal/
你想让controller包对外暴露,internal包仅内部使用,同时依赖HTTP库(比如虚构的com.http.lib):
module com.myweb.module {requires com.http.lib;
exports com.myweb.controller;
}
这样,其他模块只能访问controller包的内容,无法直接调用service或internal中的类,增强了封装性。
基本上就这些。模块机制虽然初期学习有门槛,但它让大型Java应用的结构更清晰,依赖更可控。掌握它有助于构建更健壮、可维护的系统。










