前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。
上篇文章介绍了JVM运行时数据区的一些信息,这篇文章将通过工具和字节码加深对常用的堆和虚拟机栈部分的理解。
虚拟机栈再理解
下面通过3个简单的例子再深入了解一下虚拟机栈区域。
1. 虚拟机栈的出入栈过程
这段代码很简单,在main方法中调用methodA方法、methodA调用methodB、methodB调用methodC,因为每个方法在运行期间在内存中都是以栈帧的形式表示,所以启动的时候虚拟机栈入栈过程如下:
main方法是线程中运行的,运行时先把main方法栈帧压入栈底,接着再陆续把methodA方法、methodB方法、methodC方法的栈帧压入虚拟机栈。

因为虚拟机栈后进先出,所以出栈顺序是相反的,methodC运行完出栈,接着就是methodB、methodA,直至main方法运行结束。

2. 栈帧执行流程
先看一段简单的代码:
这段代码很简单,就是a和b相加的结果乘以2,然后返回。那这段代码在JVM是怎么运行的呢。先看下栈帧的结构图

因为add是个实例方法,所以局部变量表中第0位索引的变量槽默认是用于传递方法所属对象实例的引用,在方法中可以通过关键字“this”来访问到这个隐含的参数。也可以通过javap -v FrameStacks.class 来看下局部变量表结构,可以看到第0位索引存放的是this。

再通过javap -c FrameStacks.class命令看下add方法的字节码:
先简单了解下这几个字节码的意思:
-
iconst_<n> :将一个int类型常量加载到操作数栈,n为将要操作的数值或者常量池行号 -
istore_<n> :将一个int类型数值从操作数栈存储到局部变量表,n为局部变量的位置序号 -
iload_<n> :将一个局部变量加载到操作栈,n为局部变量的位置序号 -
iadd :int类型加法指令,运算后的结果自动入操作数栈 -
imul : int类型乘法指令,运算后的结果自动入操作数栈 -
ireturn :返回
再解释一下这段字节码的执行步骤:
通过这些执行步骤可以发现,变量会频繁的出入操作数栈,一些运算操作也是在执行引擎进行的,操作数栈只是暂存变量。其实操作数栈就类似于我们说的缓存,出入栈就是删除和添加缓存,操作数栈是线程级别的缓存,随着线程的结束操作数栈也就over了。
3. 栈帧优化
先介绍一个工具JHSDB,JHSDB是一款基于服务性代理(Serviceability Agent,SA)实现的进程外调试工具。服务性代理是 HotSpot虚拟机中一组用于映射Java虚拟机运行信息的、主要基于Java语言(含少量JNI代码)实现的 API集合。通过JHSDB可以更好的理解栈帧优化。
3.1 JHSDB的启动
要使用必须要把sawindbg.dll复制一份到jre的bin目录下,我的jdk安装目录如下图,你的可能不一样:

在jdk1.8.0_152\jre\bin目录下找到sawindbg.dll文件,复制一份到jre\bin目录下。
进入jdk1.8.0_152\lib目录,通过命令行执行java -cp .\sa-jdi.jar sun.jvm.hotspot.HSDB ,执行完时候会出现如下窗口

3.2 修改上面例子的代码
修改后启动main方法
3.3 JHSDB的使用
在cmd命令行窗口输入jps 命令,找到进程ID

打开JHSDB窗口


打开之后可以看到有好几个线程启动,我们只要选择main线程就行,然后选择左上角图片是服务器按钮查看栈内存

一个栈帧的开始是从Interpreted frame部分开始的。
第一个栈帧是当前正在执行的栈帧,在这里是Thread.sleep方法的栈帧,sleep方法是native方法,因此当前是本地方法栈,也从侧面证明了Hotspot虚拟机的本地方法栈和虚拟机栈是合二为一的。
第二个是方法add方法的栈帧、第三个是main方法的栈帧,可以看到add方法栈帧的局部变量表(locals area)部分和main方法栈帧的操作数栈(expression stack)有重合,也就是蓝色方框部分,这段区域就是共享部分,也是Hotspot虚拟机对栈帧的优化。

堆区再理解
下面通过JHSDB工具来再理解一下堆区的内存布局。
新建一个类HeapObject
在idea中Edit Configurations中添加虚拟机启动参数-XX:+UseConcMarkSweepGC -XX:-UseCompressedOops -Xmx10m,如图:

-XX:+UseConcMarkSweepGC的作用是使用CMS垃圾收集器。这样能更好的查看堆的分代情况,关于CMS垃圾收集器可自行了解,这里不做过多解释。
-XX:-UseCompressedOops 禁止指针压缩,JHSDB对指针压缩存在缺陷,建议关闭指针压缩
-Xmx10m是设置堆的最大内存为10M,在这里是为了JHSDB加快在内存中搜索对象的速度
然后在通过jps命令查看HeapObject进程ID

进程id获取到之后通过JHSDB查看具体信息,在Tools -> Object Histogram中查看类的描述信息,通过类的全限定名搜索Student类。

找到之后双击查看类的描述,这里new 了两个Student对象,会看到两个对象信息。

然后通过下方的Inspect 按钮分别查看两个对象地址对应的哪个对象

从Inspector中我们可以看到
studentSun对象的内存地址是0x0000000013832558 studentArron对象的内存地址是0x0000000013400000
再在Tools -> Heap Parameters中查看堆内存分代情况

对比一下studentArron对象的内存地址是在Eden区的范围内的,所以studentArron对象在Eden区,studentSun对象内存地址在Tenured区老年代的范围内,所以studentSun在Tenured区。
为什么这两个对象不在一个区呢?
这是因为在代码中显示调用了System.gc(),studentSun对象的分代年龄变大了又因为studentSun对象的引用还在被使用,所以就把它放到了Tenured区。
从这个例子中我们可以看到Hotspot堆内存结构目前使用的是分代划分,内存空间也是连续的,并且虽然Student类对象虽然是局部变量,但是实例还是在堆区分配的。
总结
本文通过JHSDB工具和字节码层面来更深入的了解JVM运行时数据区,对于JHSDB工具和字节码也只是一个简单的使用和说明,感兴趣的可以再深入了解一下。
















