java9原定2017年3月推出,由于模块化有问题延迟到7月份,后来又推迟到9月份。万众期待的java9由于模块化的问题一再推迟,让我们来看一下什么是java9的模块化(代号Jigsaw
)。
环境准备
jdk9抢先版下载,最新版的Intellij(2017.2)已经支持java9的模块化。
什么是模块化
将一个或者多个java文件封装起来,并且提供唯一的一个接口公布出去(module-info
)。
一个模块的格式如下:
由一个或多个package
和一个module-info
组成。这样就创建了一个简单的模块,module-info
用是访问该模块的唯一入口。
module-info
每一个模块都必须在根目录下面有一个名为module-info
的java文件。该模块的信息将在module-info
中定义。
eq:
module model {
exports test;
}
我定义了一个名为model
的模块,提供了包名为test
的包(外部可以通过module-info
引入该模块,就可以访问到test
包下的所有java文件)。
module-info
中的一些常用关键字:
-
module <moduleName>
模块标记,类似class
,interface
这些后跟模块的名称。 -
exports <package>
用来定义对外公布的package
,后面跟包名,可以提供多个。 -
requires <moduleName>
用来定义当前模块需要依赖的模块,后跟模块名。 -
uses <className>
指定使用接口。 -
opens <package>
指定包下的所有java文件对外开放可以通过反射调。 -
open
作用于module
上当前模块所有公布(exports
)出去的包下面的java文件都可以通过反射调用。 -
transitive
作用于requires
将当前依赖的模块传递性暴露出去。
举个栗子
//common模块
module common {
//对外公开utils包下的所有java文件
exports utils;
}
//model模块
module model {
//对外公开entity包下的所有java文件
exports entity;
//对外公开entity包下的所有java文件
exports vos;
//依赖common模块,并且传递到外部。
requires transitive common;
}
//service模块
module service {
//依赖model模块,由于model中已经显示的声明将common传递出来。
//所以不需要再声明依赖common就可以在service中调用utils包下的java文件。
requires model;
}
上面定义了三个模块。依赖关系为common
为最底层模块,model
为中间层,service
为上层模块。如果一个模块比较复杂,根目录下有多个子package,需要公开多个package下的java文件可以显示的声明多个exports
。exports
还可以显示的指定发布到具体哪些模块中,语法如下:exports <package> to <module>,<module>,...
小结
java9去除了传统的jar引入方式,使用模块化声明式引入(我刚开始研究的时候,用的老版本的Intellij然后引入jdk9之后找不到基础jar包,编译器报找不到class错误,如果用命令行就不会出问题,这也算是刚开始研究java9的一个小坑吧)。模块化的特性:
-
声明式依赖
通过声明式的引入方式,降低了传统jar包引入的耦合关系,有点像maven的dependency(模块化没有版本的概念)。 -
增强封装
自定义封装内部实现,只需要暴露外部访问的接口类即可。具体实现外面不需要知道也访问不了(隐藏实现)。
这样在运行一些简单的demo的时候就可以节省很多空间,不需要像以前一样,把所有的jar都加载进来。这样显示的声明,需要什么引什么使java更轻量级,更方便开发。