Java SE 10 Application Class-Data Sharing 示例
Java SE 10 Application Class-Data Sharing 示例
作者:Grey
原文地址:Java SE 10 Application Class-Data Sharing 示例
Class-Data Sharing
CDS 全称 Class-Data Sharing。主要是用来在不同的 JVM 中共享 Class-Data 信息,从而提升应用程序的启动速度。
通常来说,如果要执行 class 字节码,JVM需要执行下面的一些步骤:给定一个类的名字,JVM 需要从磁盘上面找到这个文件,加载,并验证字节码,最后将它加载进来。如果 JVM 启动的时候需要加载成百上千个 class ,那么需要的就不是一个小数目了。
对于打包好的 jar 包来说,只要 jar 的内容不变,那么 jar 包中的类的数据始终是相同的。JVM 在启动时候每次都会运行相同的加载步骤。CDS 的作用就是将这些能够共享的数据归类成一个存储文件,在不同的 JVM 中共享。这样可以实现两个目标:
- 缩短JVM的启动时间。
- 减少JVM的内存占用。
当 JVM 启动时,它从文件系统中加载 JDK 类库(到JDK 8为止,从jre/lib/rt.jar文件;从JDK 9开始,从jmods目录下的jmod文件)。在这个过程中,类文件被从档案中提取出来,转换为特定架构的二进制形式,并存储在JVM进程的主内存中。
如果在同一台机器上启动了多个JVM,这个过程会重复进行。每个 JVM 在内存中保留其类库的副本。
CDS 的工作原理如下。
- 使用
-Xshare:dump命令,创建一个叫做classes.jsa的文件(JSA 代表 Java Shared Archive)。这个文件包含了当前架构下的二进制格式的完整类库。 - 当 JVM 启动时,操作系统使用 内存映射 I/O将该文件 "映射 "到 JVM 的内存中。首先,这比加载 jar 或 jmod 文件要快。其次,操作系统只将文件加载到 RAM 中一次,为每个 JVM 进程提供同一内存区域的只读视图。
Application Class-Data Sharing
Application Class-Data Sharing 扩展了 CDS,不仅可以将 JDK 类库,还可以将应用程序的类存储在 JSA 文件中,并在 JVM 进程中共享。
以下示例代码演示了App CDS 的功能。
规划项目目录
hello-cds
- src
- git
- snippets
- cds
- App.java
- Helper.java
Java 版本需要使用 JDK 10。
代码清单
App.java
package git.snippets.cds;
public class App {
public static void main(String[] args) {
new Helper().greet();
}
}
Helper.java
package git.snippets.cds;
public class Helper {
public void greet() {
System.out.println("Hello world!");
}
}
接下来在hello-cds根目录下新建target目录,并运行如下命令进行编译
javac -d target/classes src/git/snippets/cds/*.java
打包
jar cf target/helloworld.jar -C target/classes .
运行
java -cp target/helloworld.jar git.snippets.cds.App
可以看到控制台打印
Hello world!
接下来开始使用 Application CDS , 我们通过如下命令创建一个helloworld.lst
java -Xshare:off -XX:+UseAppCDS -XX:DumpLoadedClassList=helloworld.lst -cp target/helloworld.jar git.snippets.cds.App
注意。此命令仅在 OpenJDK 中有效。在 Oracle JDK 中,你会得到一个警告,说 Application CDS 是一个商业特性,你必须先解锁(用-XX:+UnlockCommercialFeatures)。所以最好使用 OpenJDK,且 JDK 版本必须是10。如果你使用Oracle JDK,用如下命令代替
java -XX:+UnlockCommercialFeatures -Xshare:off -XX:+UseAppCDS -XX:DumpLoadedClassList=helloworld.lst -cp target/helloworld.jar git.snippets.cds.App
可以用记事本打开 helloworld.lst ,发现里面包括了所有相关的class清单,这里面不仅有 JDK 的 class,也有应用程序的 class。
java/lang/Object
java/lang/String
java/io/Serializable
java/lang/Comparable
java/lang/CharSequence
java/lang/Class
....
git/snippets/cds/App
java/lang/PublicMethods$MethodList
java/lang/PublicMethods$Key
git/snippets/cds/Helper
java/lang/Shutdown
java/lang/Shutdown$Lock
接下来,我们通过 lst 问题创建 JSA 文件,
如果使用OpenJDK 10,则用如下命令
java -Xshare:dump -XX:+UseAppCDS -XX:SharedClassListFile=helloworld.lst -XX:SharedArchiveFile=helloworld.jsa -cp target/helloworld.jar
如果使用Oracle JDK 10,使用如下命令
java -Xshare:dump -XX:+UnlockCommercialFeatures -XX:+UseAppCDS -XX:SharedClassListFile=helloworld.lst -XX:SharedArchiveFile=helloworld.jsa -cp target/helloworld.jar
控制台输出如下信息
narrow_klass_base = 0x0000000800000000, narrow_klass_shift = 3
Allocated temporary class space: 1073741824 bytes at 0x00000008c0000000
Allocated shared space: 3221225472 bytes at 0x0000000800000000
Loading classes to share ...
Loading classes to share: done.
Rewriting and linking classes ...
Rewriting and linking classes: done
Number of classes 689
instance classes = 609
obj array classes = 72
type array classes = 8
Updating ConstMethods ... done.
Removing unshareable information ... done.
Scanning all metaspace objects ...
Allocating RW objects ...
Allocating RO objects ...
Relocating embedded pointers ...
Relocating external roots ...
Dumping symbol table ...
Relocating SystemDictionary::_well_known_klasses[] ...
Removing java_mirror ... done.
mc space: 5656 [ 0.1% of total] out of 65536 bytes [ 8.6% used] at 0x0000000800000000
rw space: 2079360 [ 21.4% of total] out of 2097152 bytes [ 99.2% used] at 0x0000000800010000
ro space: 3934688 [ 40.6% of total] out of 3997696 bytes [ 98.4% used] at 0x0000000800210000
md space: 6160 [ 0.1% of total] out of 65536 bytes [ 9.4% used] at 0x00000008005e0000
od space: 3413720 [ 35.2% of total] out of 3473408 bytes [ 98.3% used] at 0x00000008005f0000
total : 9439584 [100.0% of total] out of 9699328 bytes [ 97.3% used]
生成的 helloworld.jsa 文件仅 9.31 MB 大小
使用jsa文件重新运行程序
如果使用OpenJDK 10运行如下命令
java -Xshare:on -XX:+UseAppCDS -XX:SharedArchiveFile=helloworld.jsa -cp target/helloworld.jar git.snippets.cds.App
如果使用Oracle JDK 10,运行如下命令
java -Xshare:on -XX:+UnlockCommercialFeatures -XX:+UseAppCDS -XX:SharedArchiveFile=helloworld.jsa -cp target/helloworld.jar git.snippets.cds.App
控制台输出
Hello world!
源码
参考文档
Java 10 Features (with Examples)
JEP 310: Application Class-Data Sharing
Java SE 10 Application Class-Data Sharing 示例的更多相关文章
- Java SE 10 新增特性
Java SE 10 新增特性 作者:Grey 原文地址:Java SE 10 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...
- 最新版本的JDK安装和配置(Java SE 10.0.2)
1.废话少说,要么百度JDK,要么直接点传送门http://www.oracle.com/technetwork/java/javase/downloads/index.html.这里需要说的JDK包 ...
- Java SE 13 新增特性
Java SE 13 新增特性 作者:Grey 原文地址:Java SE 13 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...
- Java SE 17 新增特性
Java SE 17 新增特性 作者:Grey 原文地址:Java SE 17 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...
- Java SE 9 多版本兼容 JAR 包示例
Java SE 9 多版本兼容 JAR 包示例 作者:Grey 原文地址:Java SE 9 多版本兼容 JAR 包示例 说明 Java 9 版本中增强了Jar 包多版本字节码文件格式支持,也就是说在 ...
- Java SE 9 模块化示例
Java SE 9 模块化示例 作者:Grey 原文地址:Java SE 9 模块化示例 说明 Java SE 9引入了模块系统,模块就是代码和数据的封装体.模块的代码被组织成多个包,每个包中包含Ja ...
- Java SE 9 新增特性
Java SE 9 新增特性 作者:Grey 原文地址: Java SE 9 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new_ ...
- Java SE 8 新增特性
Java SE 8 新增特性 作者:Grey 原文地址: Java SE 8 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new_ ...
- Java SE 11 新增特性
Java SE 11 新增特性 作者:Grey 原文地址:Java SE 11 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...
随机推荐
- 关键路径 p3 清华复试上机题
关键路径 p3 清华复试上机题 题目描述 小H为了完成一篇论文,一共要完成n个实验.其中第i个实验需要a[i]的时问去完成.小H可以同时进行若干实验,但存在一些实验,只有当它的若干前置实验完成时,才能 ...
- 论文解读(KP-GNN)《How Powerful are K-hop Message Passing Graph Neural Networks》
论文信息 论文标题:How Powerful are K-hop Message Passing Graph Neural Networks论文作者:Jiarui Feng, Yixin Chen, ...
- 推荐一款M1芯片电脑快速搭建集群的虚拟机软件
虚拟机软件太多了,出名的莫过于VMware,VirutlaBox以及Parallels Desktop. 我们使用虚拟机软件一般有两种用途: 安装不同于宿主机系统的拥有用户界面的操作系统,比如Wind ...
- 如何在vscode 背景配置一个动态小女孩
D:\Microsoft VS Code\resources\app\out\vs\code\electron-browser\workbench <!-- Copyright (C) Micr ...
- SAP Web Dynpro - 教程
SAP Web Dynpro是一种标准的SAP UI技术,用于使用图形工具和与ABAP工作台集成的开发环境来开发Web应用程序. 图形工具的使用减少了实施工作,并有助于维护ABAP工作台中的组件. 本 ...
- Java获取汉字的大小写拼音码(汉字的拼音首字母)
import java.io.UnsupportedEncodingException; /** * 获取拼音码 * * @author xmj * */ public class GetPinyin ...
- 比我的脸还干的gan货——Python Flask Web 框架入门
Flask是一个轻量级的基于Python的web框架. 本文适合有一定HTML.Python.网络基础的同学阅读. 1. 简介 这份文档中的代码使用 Python 3 运行.是的,所以读者需要自己在电 ...
- IDEA快速创建maven项目
遇到问题不要急,不要怕. 一. 二. 三. 四.Finish进来之后,项目会加载一会,之后会是下面这样子. 五.继续往下面配置,建立java和resorces文件夹 六.下面配置tomcat服 ...
- Windows下maven配置环境变量
右键 "计算机",选择 "属性",之后点击 "高级系统设置",点击"环境变量",来设置环境变量,有以下系统变量需要配置: ...
- Java服务假死后续之内存溢出
一.现象分析 上篇博客说到,Java服务假死的原因是使用了Guava缓存,30分钟的有效期导致Full GC无法回收内存.经过优化后,已经不再使用Guava缓存,实时查询数据.从短期效果来看,确实解决 ...