Java性能优化权威指南第三章JVM概览第一部分

内建的JIT编译器、日渐成熟的垃圾收集器(多线程化)、不断改进的运行时环境(JVM Runtime Environment)。

HotSpot VM组件

VM运行时(Runtime)、JIT编译器(JIT Compiler)、内存管理器(Memory Manager)。

HotSpot VM的基本架构

HotSpot VM架构功能强大,可以满足高性能和高扩展性。支持HotSpot VM JIT编译器的动态优化,可以在Java应用运行时制定优化策略,并依据底层系统架构生成高效的本地机器指令。


JIT编译器(Client或Server)和垃圾收集器(Serial、Throughput、CMS或G1)都是可插拔的。
HotSpot VM运行时系统为HotSpot JIT编译器和垃圾收集器提供服务和通用API。还为VM提供启动、线程管理、JNI(Java本地接口)等基本功能。

早期的HotSpot VM(32位JVM)

早期的HotSpot VM是32位JVM,内存地址空间限制为4G,关键的是,实际Java堆的大小还进一步受限于底层操作系统。
Windows系统上HotSpot VM最大可用的Java堆大约为1.5G。
HotSpot VM在最新Linux内核上的最大可用Java堆大约为2.5G到3.0G。在较早内核版本上大约为2G。
Solaris上HotSpot VM最大可用的Java堆大约为3.3G。

实际消耗的最大内存地址空间随给定的Java应用和JVM版本而有所不同。

64位HotSpot VM

增大了Java堆,使得这些系统可以使用更多内存。

64位HotSpot VM——性能损失——压缩指针

HotSpot VM内部Java对象表示(普通对象指针,Ordinary Object Pointers,oops)长度从32位变为64位,导致CPU高速缓存行中可用的普通对象指针变少,从而降低了CPU缓存的效率。缓存效率的降低导致性能比32位JVM降低8%~15%。
最新的Java 6 HotSpot VM添加了压缩指针(Compressed oops,-XX:+UseCompressedOops开启)的新特性,使得64位Java的大尺寸堆和32位JVM的性能可以兼得。
一些Java应用在64位HotSpot VM上使用压缩指针之后,性能要好于32位VM。

64位HotSpot VM——性能提高原因

64位JVM可以使用更多的CPU寄存器,这有助于程序性能的改善。更多的CPU寄存器可以避免寄存器卸载。当活跃状态数超过CPU寄存器数,多出的活跃状态只能存放在内存中时,就会发生寄存器卸载。寄存器卸载时,某些活跃状态必须从CPU寄存器卸载到内存中,因此避免寄存器卸载可以让程序执行得更快。

压缩指针改善性能的原因

通过对齐、偏移量(Offset)将64位指针压缩成32位。
性能提高是因为使用了更小更节省空间的压缩指针而不是完整长度的64位指针,CPU缓存使用率由此得到改善。

HotSpot VM运行时责任

HotSpot VM运行时环境责任:命令行选项解析、VM生命周期管理、类加载、字节码解释、异常处理、同步、线程管理、Java本地接口、VM致命错误处理、C++(非Java)堆管理。

HotSpot VM运行时——命令行选项——使用的对象——启动器和HotSpot VM

命令行选项

  • HotSpot VM启动器使用(指定选择哪个JIT编译器、选择何种垃圾收集器)
  • 经启动器处理后传给完成启动的HotSpot VM。

HotSpot VM运行时——命令行选项——类别

  • 标准选项(Standard Option)
  • 非标准选项(Nonstandard Option)
  • 非稳定选项(Developer Option)

HotSpot VM运行时——命令行选项——标准选项

Java Virtual Machine Specification要求所有Java虚拟机都必须实现的选项,它们在发行版之间保持稳定,但也可能在后续的发行版中被废除。

HotSpot VM运行时——命令行选项——非标准选项(以-X为前缀)

不保证、也不强制所有JVM实现都必须支持,它可能未经通知就在Java SDK发行版之间发生更改。

HotSpot VM运行时——命令行选项——非稳定选项(以-XX为前缀)

为了特定需要而对JVM的运行进行校正,并且可能需要有系统配置参数的访问权限。
非稳定选项也可能不经通知就在发行版之间发生变动。

HotSpot VM运行时——命令行选项——布尔类型的选项

命令行选项用于控制HotSpot VM的内部变量,每个变量都有类型和默认值。
对于内部变量为布尔类型的选项来说,只要在HotSpot VM命令行上添加或去掉就可以控制这些变量。
对于带有布尔标记的非稳定选项来说,选项名前的+或-表示true或false,用于开启或关闭特定的HotSpot VM特性或参数。

HotSpot VM运行时——VM生命周期——启动器

HotSpot VM运行时系统负责启动和停止HotSpot VM。
启动HotSpot VM的组件是启动器:
Unix/Linux: java
Windows:java和javaw
JNI接口:JNI_CreateJavaVM启动内嵌的JVM
网络启动器:javaws(Java Web Start),Web浏览器用它来启动applet。

HotSpot VM运行时——VM生命周期——启动器启动HotSpot VM步骤

(1)解析命令行选项
启动器会直接处理一些命令行选项,例如-client-server,它们决定加载那个JIT编译器,其他参数则传给HotSpot VM。
(2)设置堆的大小和JIT编译器。
如果命令行没有明确设置堆的大小和JIT编译器(client或server),启动器则通过自动优化进行设置。自动优化的默认设定因底层系统配置和操作系统而有所不同。
(3)设定环境变量如LD_LIBRARY_PATH。
(4)如果命令行有-jar选项,启动器则从指定JAR的manifest中查找Main-Class,否则从命令行读取Main-Class。
(5)使用标准Java本地接口(Java Native Interface,JNI)方法JNI_CreateJavaVM在新创建的线程中创建HotSpot VM。
与后创建的线程相比,初始线程是启动新进程时操作系统内核分配的第一个线程。
不在初始线程中创建HotSpot VM,是为了可以对它进行定制。
(6)一旦创建并初始化好HotSpot VM,就会加载Java Main-Class,启动器也会从Java Main-Class中得到Java main方法的参数。
(7)HotSpot VM通过JNI方法CallStaticVoidMethod调用Java Main方法,并将命令行选项传给它。
至此,HotSpot VM开始正式执行命令行指定的Java程序了。

一旦Java程序或Java Main方法执行结束,HotSpot VM就必须检查和清理所有程序或者方法执行过程中生成的未处理异常。此外,方法的退出状态和程序的退出状态也必须返回给它们的调用者。
调用Java本地接口方法DetachCurrentThread将Java main方法与HotSpot VM脱离(Detached)。每次HotSpot VM调用DetachCurrentThread时,线程数就会减1,因此Java本地接口知道何时可以安全地关闭HotSpot VM,并能确保当时HotSpot VM中没有正在执行的操作,Java栈中也没有激活的Java帧。

JNI_CreateJavaVM详解

(1)确保只有一个线程调用这个方法并且确保只创建一个HotSpot VM实例。
HotSpot VM创建的静态数据结构无法再次初始化,所以一旦初始到达到某个确定点后,进程空间里就只能有一个HotSpot VM。HotSpot VM启动至此已经是无法逆转了。
(2)检查并确保支持当前的JNI版本,初始化垃圾收集日志的输出流。
(3)初始化OS模块,如随机数生成器,当前进程id、高精度计时器、内存页尺寸、保护页。
保护页是不可访问的内存页,用做内存访问区域的边界。例如操作系统常在线程栈顶压入一个保护页以保证引用不会超出栈的边界。
(4)解析传入JNI_CreateJavaVM的命令行选项,保存以备将来使用。
(5)初始化标准的Java系统属性,例如java.version、java.vendor、os.name等。
(6)初始化支持同步、栈、内存和安全点页的模块。
(7)加载libzip、libhpi、libjava及libthread等库。
(8)初始化并设置信号处理器(Signal Handler)。
(9)初始化线程库。
(10)初始化输出流日志记录器(Logger)。
(11)如果用到Agent库(hprof、jdi),则初始化并启动。
(12)初始化线程状态(Thread State)和线程本地存储(Thread Local Storage),它们存储了线程私有数据。
(13)初始化部分HotSpot VM全局数据,例如事件日志(Event Log),OS同步原语、perfMemory(性能统计数据内存),以及chunkPool(内存分配器)。
(14)至此,HotSpot VM可以创建线程了。创建出来的Java版main线程被关联到当前操作系统的线程,只不过还没有添加到已知线程列表中。
(15)初始化并激活Java级别的同步。
(16)初始化启动类加载器(Bootclassloader)、代码缓存、解释器、JIT编译器、JNI、系统词典(System Dictionary)及universe(一种必备的全局数据结构集)。
(17)现在,添加Java主线程到已知线程列表中。检查universe是否正常。创建HotSpot VMThread,它执行HotSpot VM所有的关键功能。同时发出适当的JVMTI事件,报告HotSpot VM的当前状态。
(18)加载和初始化以下Java类:java.lang.String、java.lang.System、java.lang.Thread、java.lang.ThreadGroup、java.lang.reflect.Method、java.lang.ref.Finalizer、java.lang.Class以及余下的Java系统类。此时,HotSpot已经初始化完毕并可使用,只是功能还不完备。
(19)启动HotSpot VM的信号处理线程,初始化JIT编译器并启动HotSpot编译代理线程。启动HotSpot VM辅助线程(如监控线程和统计抽样器)。至此,HotSpot VM已功能完备。
(20)最后,生成JNIEnv对象返回给调用者,HotSpot则准备响应新的JNI请求。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • 这篇文章是我之前翻阅了不少的书籍以及从网络上收集的一些资料的整理,因此不免有一些不准确的地方,同时不同JDK版本的...
    高广超阅读 15,572评论 3 83
  • 《深入理解Java虚拟机》笔记_第一遍 先取看完这本书(JVM)后必须掌握的部分。 第一部分 走近 Java 从传...
    xiaogmail阅读 5,070评论 1 34
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,220评论 11 349
  • 已经不是每天一更了,这个礼拜在匆忙开学匆忙备战匆忙大姨妈中度过。。想想自己的记性真是不太好,甚至回忆不起来过去两天...
    斯普特尼克sptnk阅读 222评论 0 0
  • 一纸夜的宁静 一笔心的清欢 抒一首小诗 不小心飞落酒里 溶入血液 深彻心扉 灵魂里有了诗的香气 入一怀盈盈闪闪的思...
    紫色靈魂阅读 273评论 0 0