Module是Java9引入的新功能,旨在解决Jar包洪灾问题,在Java9之前,类的访问只能由public/protected/default/private加包来控制,module相当于在public上再加一层审控制。这里有两篇文章写得很详细,我就不在这里讲它是怎么使用以及它的来由了。
Java 9 Module的特性以及解决的问题blog.csdn.netmodule由module-info.java来定义,这个文件最后会编译成一个module-info.class文件,但是没有类申明,只有Module的声明,见下面的注释:
module
Module的解析从com.sun.tools.javac.comp.Modules.initModulesnitModules(List<JCCompilationUnit> trees)开始。
然后就干了以下同几件事:
- 如果trees里没有module-info.java的编译JCCompilationUnit,就去找包根目录下的是不是有module-info.java,然后通过一个Completor回调做下一个步骤的事件。
- 如果trees里含有module-info.java的编译JCCompilationUnit,就用它创建一个ModuleSymbol,然后负值给trees里的其它成员。
- 通过module.Completor做以下几件事:
- 加载requires里早申明的module里exports的所有包,这就是这个Module所能用到的包。
- 把这个包以及它所exports里的类放到Symtab里的。(Symtab是事个编译器所有类型所在的一个HashMap)
- 检查是不是有循环引用的module.
- 如果没找到module-info.java,那么这个module就一个unnamed的module,它所有的类可见与Java8之前的一样。
这部分其实也没有什么意思 ,但是引入了两个重要的概念可以在这里说一下:
JCTree.Visitor类,这是一个经典的Visitor模式,主要是为之后各种功能实现对JCTree的访问提供默认实现。关于Visitor模式,下面这篇讲得挺好的,我在这里就不赘述了。
设计模式(Java)-Visitor模式_Java_zoinsung_lee的博客-CSDN博客blog.csdn.netModules里有两个内部类继承了JCTree.Visitor来集中实现对每一种AST节点的处理。Modules$ModuleVisitor和Modules$UsesProvidesVisitor,这两个类都只对Module相关的JCTree子类处理,如下图所示:
Modules本身也扩展了JCTree.Visitor但是没有扩展任何方法,所以它不会对任何树的结点处理。
另一个是com.sun.tools.javac.Symbol$Completer,它是在一个Symbol的一个completer属性上的一个回调。在不同的阶段赋值不同的Completer实现来做不同的事情。
2020年1月马智出了一本专门讲这个的书《深入解析Java编译器:源码剖析与实例详解》,基本上覆盖了所有的要点,所以专栏暂时不更新了,如果需要了解这方面的知识请翻阅这本书。