前言
截止到目前jdk的版本已經更新到10了,雖然java9的生命周期才半年,但是我認為這個版本帶來的變革是不可磨滅的,它是第一次深層次的針對架構以及依賴上的革新。下面我們就來學習一下。
模塊化的功能有幾個目的:
- 讓java的se程序更加容易輕量級部署
- 改進組件間的依賴管理,引入比jar粒度更大的module
- 改進性能和安全性
- 如果用更加簡單解釋,那就是"解決classpath地獄問題,改進部署能力"。module的內容比較多,為了由淺入深,我按照一些問題和我的理解來介紹模塊化。
一、模塊化項目構建
其實模塊化本身不難理解,我們先前使用maven或者gradle就構建過多模塊的項目。那么我們在java9里依然可以照貓畫虎來構建一下我們的模塊化項目工程。如圖所示:
注意以下幾點:
1.請在每個模塊下創建一個叫做module-info.java的模塊化描述文件
2.在idea里配置一下模塊依賴,在這里我們的project.portal模塊如果依賴student.service模塊,我們可以這么來設置:
找到這個選項圖標:,然后這樣設置來添加依賴:
如果需要設置其他項目的依賴項,也請按照此方式設置。
二、實現步驟
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(); } } |
我們運行后既可以拿到對應的結果:
三、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修飾符會導致依賴于當前模塊的其他模塊具有隱式依賴性,請看下圖:
在這里我們可以看看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