主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
Java

Java 基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

Java 控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

Java 面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

Java 内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

Java 文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

Java 错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

Java 多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

Java 同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

Java 网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

Java 集合

Java 集合框架
Java 集合接口

Java 接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

Java 数据结构

Java 数据结构
Java 枚举接口

Java 集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

Java 高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

Java 杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

集合

Java 集合框架
Java 集合接口

接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

数据结构

Java 数据结构
Java 枚举接口

集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

Java 多版本发布 JAR


上一章 下一章

多版本发布 JAR(Multi-release JAR)特性是在 Java 9 中引入的。这一特性允许使用多个版本的类来适应 Java 的不同版本。例如,许多第三方库或框架都是基于 Java 构建的。随着 Java 语言的不断演进,每次主要版本更新都会给语言添加许多新功能。为了兼容这些新功能,第三方库/框架需要重写其代码库,这让团队非常不愿意采用新功能。这对它们来说是一个阻碍,使得它们难以迁移到 Java 的新版本。

为了应对维护同一文件的多个源代码版本或特定平台版本的问题,引入了多版本发布 JAR 特性。考虑以下示例:

典型的 JAR 文件结构

一个典型的 JAR 文件会在根级别包含所有的类。

jar root
   - Calculator.class
   - Util.class
   - Math.class
   - Service.class

如果有一个针对 Java 9 特性的 Util 类,那么这个 JAR 文件就不能在 Java 8 或更低版本的环境中使用。

多版本发布 JAR 解决方案

在多版本发布的 JAR 文件中,格式已经被增强,可以维护和使用适用于不同平台的不同版本的 Java 类或资源。在 JAR 文件中,MANIFEST.MF 文件在其主要部分有一个条目 Multi-Release: true。META-INF 目录也包含一个名为 versions 的子目录,其下级目录(以 9 表示 Java 9)存储版本特定的类和资源文件。

利用 MANIFEST.MF 文件,我们可以将 Java 9 或更高版本特定的类放置在不同的位置,如下所示:

Java Multi-Release Jar Files Directory Structure
jar root
   - Calculator.class
   - Util.class
   - Math.class
   - Service.class
   META-INF
      - versions
      - 9
         - Util.class
         - Math.class
      - 10
         - Util.class
         - Math.class

如果 JRE 不支持多版本发布的 JAR 文件,那么它会选择根级别的类进行加载和执行;否则,将加载特定版本的类。例如,如果上述 JAR 文件在 Java 8 中使用,那么将会使用根级别的 Util.class。如果在同一 JRE 下执行,JRE 是 Java 9,那么将会选择 Java 9 版本特定的类,依此类推。这样,第三方库/框架可以在不改变其针对较低版本编写的源代码的情况下支持新功能。

创建和使用多版本发布的 JAR 文件

在这个例子中,我们将创建并使用一个多版本发布的 JAR 文件,包含两个版本的 Tester.java 文件,一个针对 JDK 7,另一个针对 JDK 9,并在不同版本的 JDK 上运行它。

以下是创建和使用多版本发布的 JAR 文件的步骤:

步骤 1:创建针对 Java 7 的 Java 类

让我们创建一个具有 Java 7 特定代码和特性的 Java 类,这些在 Java 9 之前是不可用的。在这个例子中,我们仅仅是打印一条消息来展示这个特性的使用。

创建一个文件夹 c:/test/java7/com/tutorialspoint。创建 Tester.java 并包含以下内容:

Tester.java

这是一个简单的代码,在程序执行时打印一条 Java 7 特定的消息。
package com.tutorialspoint;

public class Tester {
   public static void main(String[] args) {
      System.out.println("Inside java 7");
   }
}

步骤 2:创建针对 Java 9 的 Java 类

让我们创建一个具有 Java 9 增强特性和功能的 Java 类,这些在 Java 9 之前是不可用的。在这个例子中,我们同样是打印一条消息来展示这个特性的使用。

创建一个文件夹 c:/test/java9/com/tutorialspoint。创建 Tester.java 并包含以下内容:

Tester.java

这也是一段类似的代码,在程序执行时打印一条 Java 9 特定的消息。
package com.tutorialspoint;

public class Tester {
   public static void main(String[] args) {
      System.out.println("Inside java 9");
   }
}

为了使用多版本发布的 JAR 特性,必须确保两个类的签名相同。公共接口如方法签名在两个类中应该是相同的。

步骤 3:使用目标版本编译

C:\test > javac --release 9 java9/com/tutorialspoint/Tester.java

C:\JAVA > javac --release 7 java7/com/tutorialspoint/Tester.java

步骤 4:创建一个多版本发布的 JAR 文件

C:\JAVA > jar -c -f test.jar -C java7 . --release 9 -C java9 .
Warning: entry META-INF/versions/9/com/tutorialspoint/Tester.java,
   multiple resources with same name

步骤 5:使用 JDK 7 运行 JAR 文件

C:\JAVA > java -cp test.jar com.tutorialspoint.Tester
Inside Java 7

步骤 6:使用 JDK 9 运行 JAR 文件

C:\JAVA > java -cp test.jar com.tutorialspoint.Tester
Inside Java 9

结论

我们可以看到,通过多版本发布的 JAR 特性,我们可以在没有向后兼容性问题的情况下创建一个类的多个版本。我们展示了同样的命令可以用于执行一个类。根据列在 META-INF 文件中的 JRE 版本,将选择相应的类。对于较低版本的 JRE,如果它们不支持多版本发布的 JAR 文件,将选择根级别的类而不是版本特定的类。最后,版本化类的公共方法签名应该相同,这样此功能才能完美工作。

上一章 下一章
阅读号二维码

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图