美文网首页
Java9模块化学习

Java9模块化学习

作者: rubywang08 | 来源:发表于2018-11-12 10:10 被阅读0次

1. 什么是模块化

本质上,模块化就是将系统分解成独立且相互连接的模块的行为。java9模块除了包含代码外,一个重要特征是增加了描述模块的文件:module-info.java

Q:为什么这么做?-- 减少复杂性

Project Jigsaw

Jigsaw 项目从2008就开始对JDk模块化进行探索,2014年开始进行设计和实现。

2. 模块化JDK

JDK由90个左右的平台模块组成,每个模块都是一个定义好的功能块,下图显示了部分模块的子集以及依赖关系:


image.png
  1. 顶部的两个重要模块:java.se.ee,java.se,主要用于对其他模块进行逻辑分组,属于聚合器模块。

  2. 每个模块都隐式依赖于java.base,因为该模块公开了java.lang, java.util之类的包。

  3. 模块之间依赖关系没有循环,java模块系统不允许编译时存在模块循环依赖。

    列出所有JDK模块: java --list-modules

    查看模块定义:java --describe-module java.rmi

3.模块如何定义

模块描述文件module-info.java:

module com.test{// 模块名称
    requires java.sql;  //依赖java.sql模块,普通依赖
    requires transitive java.xml;  //依赖java.xml,可传递依赖
    exports com.my.test1;  //导出com.my.test1包到其他模块
    exports com.my.test2 to xx.xx.xx; //限制导出,导出com.my.test2到指定模块
}
  • 可读性:模块A依赖模块B,意味着对模块B可读,也就是可以访问模块B导出包中的类型。普通依赖可读性是不可传递的。
  • 可访问性:public、protected、private等访问修饰符的可访问性规则依旧适用。

4.模块Demo

单个模块

创建模块

编译

mkdir -p mods/helloworld
javac -d mods/helloworld helloworld/src/helloworld/module-info.java helloworld/src/helloworld/com/module/helloworld/HelloWorld.java

打包

mkdir mlib
方式1:jar -cfe mlib/helloworld.jar com.module.helloworld.HelloWorld -C mods/helloworld .
方式2:jar --create --file=mlib/com.greetings.jar --main-class=com.module.helloworld.HelloWorld -C mods/helloworld .

运行

1.以编译后的文件运行模块:

java --module-path mods --module helloworld/com.module.helloworld.HelloWorld

2.以模块化jar包运行模块:

java -p mlib -m helloworld

多个模块

模块间的引用

用Maven如何配置多个模块

1.为每个模块添加maven依赖,maven只是配置模块路径。

2.pom文件中配置编译器插件为java9或以上

 <build>
    <plugins>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-compiler-plugin</artifactId>
        <version>3.7.0</version>
        <configuration>
          <source>10</source>
          <target>10</target>
        </configuration>
      </plugin>
    </plugins>
  </build>

3.给每个模块添加module-info.java,并将依赖项作为requires语句添加。

模块定义限制

模块解析是一个递归的过程,从根模块开始解析->requires/requires transitive所需的模块->循环直到没有其他任何依赖项。

  • 模块名称必须唯一

  • Java模块系统不支持编译期循环依赖

  • 只有一个模块可以导出一个给定的包

  • 模块不能定义版本,版本信息必须存储在模块之外。如Maven通过pom文件选择正确的模块版本并设置到模块路径上

image2.PNG

5. 服务

Java模块系统的服务机制可以共享公共接口,并将实现代码强封装到未导出的包中,实现服务提供者和消费者的真正解耦。

module api {
    exports com.calculate; //api模块导出服务接口
}
//Caltulator接口代码
public interface Calculator {
    String getName();
    BigDecimal calculate(BigDecimal v1, BigDecimal v2);
}
module provider1 {
    requires api;
    provides com.calculate.Calculator with com.provider.Sum;//该模块提供了Calculator接口的一个实现,并且不导出实现类
}
//Sum实现类代码
public class Sum implements Calculator {
     ......
}
module com.consumer{
    requires api;
    uses com.calculate.Calculator; //该模块想要消费Calculator的实现
}
消费端代码:
public static void main(String[] args) {
        ServiceLoader<Calculator> services = ServiceLoader.load(Calculator.class);
        for (Calculator cal : services) {
            cal.getName();
        }
  }
  • uses子句不要求Calculator实现在编译期间可用,消费者和提供者在运行时才被绑定在一起,由此实现了消费者和提供者的完全解耦,提供了不用重新编译、打包的可扩展性

  • 消费者使用ServiceLoader获取提供者实现对代码是有一定侵入性的,有两种改进的方式:

    • 在api模块接口中添加一个静态工厂方法,该方法返回所有提供者实现

      module api1 {
          exports com.calculate; //api模块导出服务接口
           uses com.calculate.Calculator; //该模块想要消费Calculator的实现 
      }
      //api模块接口代码
      public interface Calculator {
          String getName();
          BigDecimal calculate(BigDecimal v1, BigDecimal v2);
          static Iterable<Calculator> getCalculator(){
              return ServiceLoader.load(Calculator.class);
          }
      }
      
      module com.consumer1{
          requires api;
      }
      消费端代码:
      public static void main(String[] args) {
          Iterable<Calculator> services = Calculator.getCalculator();
        }
      
    • 使用开放模块和开放包,消费者通过@Inject依赖注入服务提供者类。

      open module api{// 开放模块中的所有包,任何模块都可以反射访问导出、未导出的类型的私有域
          exports com.api;// 未导出包在编译时依旧是不可访问的
      }
      module api{
          exports com.api;
          opens com.api;
      }
      

6.现有代码使用Java9

将现有代码迁移到Java9可以分两步:

在Java9上构建和运行现有代码,无需模块化

  • Java9有一个特殊的未命名模块(unnamed module),该模块可以读取其他所有的模块。在模块之外编译和加载的代码都在未命名模块中。

  • 由于JDK是模块化的,对类进行了强封装,通过反射访问这些类型时,会有警告。

    (java9中exports导出的公共类型,反射访问该类型的私有域是禁止的,为了保持兼容性,默认运行时访问可以通过java选项--illegal-access=值,控制,默认值为permit )

  • 编译和运行未命名模块时,以java.se作为根模块,如果程序使用了java.se.ee下的模块,会报错。可以使用--add-modules添加相关模块。

image3.PNG
  • 如果代码使用了已经删除的类型,改代码。可以使用JDK附带的一个工具jdeps查找在使用的已经被删除或封装的JDK类型。

将代码模块化

其实规模比较小的程序,还有比较老的系统(没有太多更新,只是维护改bug)没必要模块化。程序规模比较大而且更新比较多的话,可以考虑迁移,因为模块化是可以提高可维护性和可重用性的。

将现有代码模块化最大的问题:如何迁移第三方库,大部分第三方库都只是普通Jar文件,不是模块。

  • 创建自动模块:将现有jar文件放到模块路径,不用改变任何内容,就可以创建一个自动模块

    • 自动模块会导出所有包
    • requires transitive 所有其他已解析的模块
  • 添加该第三方库的包到被依赖的模块描述文件中

总结

模块化的好处

  • 模块描述文件,编译时就能检查模块所有依赖,减少运行时错误
  • 强封装,模块需要显示的配置向其他模块导出的内容,内部实现细节完全不可见
  • 提高可维护性
  • 提高安全性

相关文章

网友评论

      本文标题:Java9模块化学习

      本文链接:https://www.haomeiwen.com/subject/afyzaftx.html