Java9新特性之Module模块化编程示例演绎


Posted in Java/Android onMarch 16, 2022

 

我计划在后续的一段时间内,写一系列关于java 9的文章,虽然java 9 不像Java 8或者Java 11那样的核心java版本,但是还是有很多的特性值得关注。期待您能关注我,我将把java 9 写成一系列的文章,大概十篇左右,本文是第8篇。

 

在Java 9版本中Java 语言引入了一个非常重要的概念:模块(module)。如果对javascript代码模块化管理比较熟悉的小伙伴,看到Java 9的模块化管理,应该有似曾相识的感觉。

一、什么是Java module?

与Java 中的package有些类似,module引入了Java代码分组的另一个级别。每个这样的分组(module)都包含许多子package包。通过在一个模块的源代码文件package的根部,添加文件module-info.java来声明该文件夹及其子文件夹为一个模块。该文件语法如下:

module xxx.yyy{
  ....
 }

其中xxx.yyy是模块module声明的名称,不是package名称。

二、模块导出package

文件module-info.java可以指定该模块下面的哪些package对外可见、可访问。通过一个新的关键字exports来实现该功能。

module xxx.yyy{
  exports com.zimug.java9;
 }

com.zimug.java9代表一个package。

需要注意的是:即使给定package包中的类是public的,如果未通过’exports’显式导出其程序包,则它们在模块外部也是不可见的(在编译时和运行时都是如此)。

三、模块导入package

如果另一个模块想要使用被导出的package包中的类,可以用requires关键字在其module-info.java文件中来导入(读取)目标模块的package包。

module def.stu{ requires xxx.yyy;}

四、Java module的意义

在笔者看来,Java 9引入module 模块化管理系统,更多的是从安全性的角度考虑。Java 代码中90%以上的漏洞都是由反射和访问权限控制粒度不足引起的,Java 9的模块化系统正好能解决这个问题。Java 9 module提供另一个级别的Java 代码可见性、可访问性的控制。

比如说:我们都知道当一个class被修饰为private的时候,意味着这个类是内部类。对于顶级类(外部类)来说,只有两种修饰符:public和默认(default)。这也就意味着一个问题,有些public class我们本来是打算在jar包定义的范围内使用的,但是结果却是任何引入了这个jar的项目都可以使用这个jar里面所有的public class代码。

也就是我们的原意是在有限范围内提供公开访问,结果却是无限制的对外公开。在引入Java 9模块化之后,可以实现有限范围内的代码public访问权限,将代码公开区分为:模块外部有限范围的公开访问模块内部的公开访问

五、实例

在此示例中,我将创建两个模块“ common.widget”和“ data.widget”,并将它们放置在单个文件夹“ modules-examples/src”下。文件“ module-info.java”将放置在每个模块的根文件夹下。
文件及目录格式如下:

D:\modules-example>tree /F /A
\---src
    +---common.widget
    |   |   module-info.java
    |   |   
    |   +---com
    |   |   \---zimug
    |   |           RendererSupport.java
    |   |           
    |   \---org
    |       \---jwidgets
    |               SimpleRenderer.java
    |               
    \---data.widget
        |   module-info.java
        |   
        \---com
            \---example
                    Component.java

第一个模块

本代码文件目录:

modules-example/src/common.widget/org/jwidgets/SimpleRenderer.java。

这个package在后文中没有被exports。

package org.jwidgets;
public class SimpleRenderer {
  public void renderAsString(Object object) {
      System.out.println(object);
  }
}

本代码文件目录:

modules-example/src/common.widget/com/zimug/RendererSupport.java。

这个package在后文中被exports了。

package com.zimug;
import org.jwidgets.SimpleRenderer;
public class RendererSupport {
  public void render(Object object) {
      new SimpleRenderer().renderAsString(object);
  }
}

模块导出,本代码文件目录:modules-example/src/common.widget/module-info.java。只导出com.zimug包,没有导出 org.jwidgets包。导出的模块名称为common.widget

module common.widget{
  exports com.zimug;
}

第二个模块

模块导入common.widget,本代码文件目录:modules-example/src/data.widget/module-info.java

module data.widget {
  requires common.widget;
}

使用导入模块common.widget中的package:com.zimug。本代码文件路径:

modules-example/src/data.widget/com/example/Component.java

package com.example;
import com.zimug.RendererSupport;
public class Component {
  public static void main(String[] args) {
      RendererSupport support = new RendererSupport();
      support.render("Test Object");
  }
}

正常编译执行,结果如下:

Test Object

尝试使用未被exports的package代码

由于包“ org.jwidgets”尚未通过“ common.widget”模块导出,因此另一个模块“ data.widget”无法使用该package包下的类SimpleRenderer。我们做一个反例,看看会发生什么:

package com.example;
import org.jwidgets.SimpleRenderer;
public class Component {
  public static void main(String[] args) {
    SimpleRenderer simpleRenderer = new SimpleRenderer(); 
    simpleRenderer.renderAsString("Test Object");
  }
}

编译报错信息如下:

D:\modules-example\src\data.widget\com\example\Component.java:3: error: package org.jwidgets is not visible
import org.jwidgets.SimpleRenderer;         ^
  (package org.jwidgets is declared in module common.widget, which does not export it)
1 error

正如我们所看到的,未被exports的package下面的class即使是public的也不能被访问。

以上就是Java9新特性Module模块化编程示例演绎的详细内容,更多关于Java9新特性Module模块化编程的资料请关注三水点靠木其它相关文章!

Java/Android 相关文章推荐
Java Dubbo框架知识点梳理
Jun 26 Java/Android
SpringBoot实现异步事件驱动的方法
Jun 28 Java/Android
在Spring-Boot中如何使用@Value注解注入集合类
Aug 02 Java/Android
Java 超详细讲解IO操作字节流与字符流
Mar 25 Java/Android
详解Flutter和Dart取消Future的三种方法
Apr 07 Java/Android
零基础学java之方法的定义与调用详解
Apr 10 Java/Android
Java 使用类型为Object的变量指向任意类型的对象
Apr 13 Java/Android
JAVA 线程池(池化技术)的实现原理
Apr 28 Java/Android
Spring 使用注解开发
May 20 Java/Android
springboot为异步任务规划自定义线程池的实现
Jun 14 Java/Android
Java服务调用RestTemplate与HttpClient的使用详解
Jun 21 Java/Android
Android移动应用开发指南之六种布局详解
Sep 23 Java/Android
JVM的类加载器和双亲委派模式你了解吗
Java生成日期时间存入Mysql数据库的实现方法
Mar 03 #Java/Android
Java设计模式之享元模式示例详解
解析探秘fescar分布式事务实现原理
关于ObjectUtils.isEmpty() 和 null 的区别
Feb 28 #Java/Android
java objectUtils 使用可能会出现的问题
Feb 28 #Java/Android
JVM之方法返回地址详解
Feb 28 #Java/Android
You might like
小文件php+SQLite存储方案
2010/09/04 PHP
ThinkPHP框架任意代码执行漏洞的利用及其修复方法
2014/07/04 PHP
js判断为空Null与字符串为空简写方法
2014/02/24 Javascript
浅谈javascript中return语句
2015/07/15 Javascript
Jquery插件之Fancybox丰富的弹出层效果附源码下载
2015/12/02 Javascript
JQuery中解决重复动画的方法
2016/10/17 Javascript
JS正则表达式判断有效数实例代码
2017/03/13 Javascript
JS中使用正则表达式g模式和非g模式的区别
2017/04/01 Javascript
快速掌握jquery分页插件jqPaginator的使用方法
2017/08/09 jQuery
Javascript刷新页面的实例
2017/09/23 Javascript
如何重置vue打印变量的显示方式
2017/12/06 Javascript
Bootstrap Table中的多选框删除功能
2018/07/15 Javascript
koa2使用ejs和nunjucks作为模板引擎的使用
2018/11/27 Javascript
JS实现的合并两个有序链表算法示例
2019/02/25 Javascript
jQuery+ajax实现用户登录验证
2020/09/13 jQuery
python实现的登陆Discuz!论坛通用代码分享
2014/07/11 Python
ubuntu中配置pyqt4环境教程
2017/12/27 Python
python 实现调用子文件下的模块方法
2018/12/07 Python
Python3.5基础之函数的定义与使用实例详解【参数、作用域、递归、重载等】
2019/04/26 Python
Django实现微信小程序的登录验证功能并维护登录态
2019/07/04 Python
详解python中的time和datetime的常用方法
2019/07/08 Python
python对Excel按条件进行内容补充(推荐)
2019/11/24 Python
Python 实现Serial 与STM32J进行串口通讯
2019/12/18 Python
Python Opencv轮廓常用操作代码实例解析
2020/09/01 Python
HTML5拖拽功能实现的拼图游戏
2018/07/31 HTML / CSS
英国最大的在线亚洲杂货店:Red Rickshaw
2020/03/22 全球购物
市场营销管理制度
2014/01/29 职场文书
软件部经理岗位职责范本
2014/02/25 职场文书
委托书模板
2014/04/04 职场文书
《北大荒的秋天》教学反思
2014/04/14 职场文书
热血教师观后感
2015/06/10 职场文书
离婚协议书范文2016
2016/03/18 职场文书
使用 Apache Superset 可视化 ClickHouse 数据的两种方法
2021/07/07 Servers
Vue Element-ui表单校验规则实现
2021/07/09 Vue.js
Python语法学习之进程的创建与常用方法详解
2022/04/08 Python
Redis实现主从复制方式(Master&Slave)
2022/06/21 Redis