java9学习笔记之模块化详解

时间:2022-04-20 05:47:46

前言

  截止到目前jdk的版本已经更新到10了,虽然java9的生命周期才半年,但是我认为这个版本带来的变革是不可磨灭的,它是第一次深层次的针对架构以及依赖上的革新。下面我们就来学习一下。

模块化的功能有几个目的:

  • 让java的se程序更加容易轻量级部署
  • 改进组件间的依赖管理,引入比jar粒度更大的module
  • 改进性能和安全性
  • 如果用更加简单解释,那就是"解决classpath地狱问题,改进部署能力"。module的内容比较多,为了由浅入深,我按照一些问题和我的理解来介绍模块化。

一、模块化项目构建

  其实模块化本身不难理解,我们先前使用maven或者gradle就构建过多模块的项目。那么我们在java9里依然可以照猫画虎来构建一下我们的模块化项目工程。如图所示:

java9学习笔记之模块化详解

注意以下几点:

  1.请在每个模块下创建一个叫做module-info.java的模块化描述文件

  2.在idea里配置一下模块依赖,在这里我们的project.portal模块如果依赖student.service模块,我们可以这么来设置:

  找到这个选项图标:,然后这样设置来添加依赖:  

   如果需要设置其他项目的依赖项,也请按照此方式设置。

java9学习笔记之模块化详解

二、实现步骤

2.1、student.service模块

2.1.1、编写studentservice的module-info.java

示例代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
import com.bdqn.lyrk.student.service.secondstudentservice;
import com.bdqn.lyrk.student.service.api.istudentservice;
/**
 * 模块化描述类,统一建立在各个模块的源文件根目录 名字为:module-info.java
 * 模块化常见的语法结构:
 *
 * import xxxx.xxxx;
 * ....
 *
 * [open] module 模块名 {
 * requires [static|transitive] 模块名;
 * exports 包名 [to 模块名]
 * providers 接口名 with [接口实现类,....]
 * uses 接口名
 *
 * }
 *
 *
* @author chen.nie
* @date 2018/4/18
**/
module student.service {
 
 exports com.bdqn.lyrk.student.service.api;
 provides istudentservice with secondstudentservice;
}

2.1.2、定义接口

?
1
2
3
4
package com.bdqn.lyrk.student.service.api;
public interface istudentservice {
 void study();
}

2.1.3、定义实现类

?
1
2
3
4
5
6
7
8
9
package com.bdqn.lyrk.student.service;
import com.bdqn.lyrk.student.service.api.istudentservice;
 
public class secondstudentservice implements istudentservice {
 @override
 public void study() {
 system.out.println("second study");
 }
}

2.2、project.portal 模块

2.2.1、编写module-info.java

?
1
2
3
4
5
import com.bdqn.lyrk.student.service.api.istudentservice;
module project.portal {
 uses istudentservice;
 requires transitive student.service;
}

2.2.2、编写main方法

?
1
2
3
4
5
6
7
8
9
package com.bdqn.lyrk.portal;
import com.bdqn.lyrk.student.service.api.istudentservice;
import java.util.serviceloader;
public class main {
 public static void main(string[] args) {
 serviceloader<istudentservice> studentservices = serviceloader.load(istudentservice.class);
 studentservices.findfirst().get().study();
 }
}

我们运行后既可以拿到对应的结果:

java9学习笔记之模块化详解

三、module-info.java文件常见配置

3.1、关于open关键字

  open:该关键字如果加载模块上,那么通过exports的导出包下的类可见度是最高的,我们可以通过反射的方式来创建对对象和访问属性。

3.2、关于exports关键字

  当我们定义好模块后,我们可以指定该模块下的哪些包可以被其他模块所访问,exports关键字就起到该作用。我们也可以配合to来指定哪些模块可以访问该包的内容

  语法 exports 包名 [to] 模块名

?
1
2
exports <package>;
exports <package> to <module1>, <module2>...;

3.3、opens关键字

    opens类似于open,如果open关键字加在module上,那么模块里默认导出的exports包都为open形式的

?
1
2
3
4
module n {
 exports com.jdojo.claim.model;
 opens com.jdojo.claim.model;
}

3.4、requires关键字

       该关键字声明当前模块与另一个模块的依赖关系。有点类似于maven中的dependecies。

?
1
2
3
4
requires <module>;
requires transitive <module>;
requires static <module>;
requires transitive static <module>;

  require语句中也可以加静态修饰符,这样的话表示在编译时的依赖是强制的,但在运行时是可选的。require语句中的transitive修饰符会导致依赖于当前模块的其他模块具有隐式依赖性,请看下图:

java9学习笔记之模块化详解

       在这里我们可以看看java.se模块下的module-info.class文件:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
/*
 * copyright (c) 2014, oracle and/or its affiliates. all rights reserved.
 * oracle proprietary/confidential. use is subject to license terms.
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 *
 */
 
/**
 * defines the core java se api.
 * <p>
 * the modules defining the corba and java ee apis are not required by
 * this module, but they are required by the
 * <a href="java.se.ee-summary.html">{@code java.se.ee}</a> module.
 *
 * <dl>
 * <dt class="simpletaglabel" style="font-family:'dejavu sans', arial, helvetica, sans serif">optional for the java se platform:</dt>
 * <dd>
 * <a href="../specs/jni/index.html">java native interface (jni)</a><br>
 * <a href="../specs/jvmti.html">java virtual machine tool interface (jvm ti)</a><br>
 * <a href="../specs/jdwp/jdwp-spec.html">java debug wire protocol (jdwp)</a><br>
 * </dd>
 * </dl>
 *
 * @modulegraph
 * @since 9
 */
module java.se {
 requires transitive java.compiler;
 requires transitive java.datatransfer;
 requires transitive java.desktop;
 requires transitive java.instrument;
 requires transitive java.logging;
 requires transitive java.management;
 requires transitive java.management.rmi;
 requires transitive java.naming;
 requires transitive java.prefs;
 requires transitive java.rmi;
 requires transitive java.scripting;
 requires transitive java.security.jgss;
 requires transitive java.security.sasl;
 requires transitive java.sql;
 requires transitive java.sql.rowset;
 requires transitive java.xml;
 requires transitive java.xml.crypto;
}

  此时我们只要requires java.se,那么该模块下的所有依赖我们就间接的引入了

3.5、uses与provider关键字

       java允许使用服务提供者和服务使用者分离的服务提供者机制。 jdk 9允许使用语句(uses statement)和提供语句(provides statement)实现其服务。使用语句可以指定服务接口的名字,当前模块就会发现它,使用 java.util.serviceloader类进行加载。代码请参考前面的例子,注意:provider提供的类必须在同一个模块下,当前不能引用其他模块的实现,比如说:前面的例子studentserviceimpl只能存在student.service模块下,student.service模块provider其他的模块下的接口实现是不允许的。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。

原文链接:http://www.cnblogs.com/niechen/p/8875372.html