三、虚拟机栈
-
虚拟机栈的概述、常见的异常和如何设置大小
1、Java虚拟机栈,早期也叫Java栈,每个线程工作的时候都会在运行时数据区中创建一个虚拟机栈,内部是一个个栈帧,每个栈帧对应Java程序里的一个方法,生命周期和线程保持一致。
2、作用:主管Java程序的运行,保存方法的局部变量和部分结果,以入栈、出栈的方式,参与方法的调用和返回。
3、在虚拟机栈中,最常见的异常就是栈溢出(StackOverFlowError:在虚拟机栈是固定大小的情况下,当线程请求分配的栈容量超过虚拟机栈的最大容量,就会报栈溢出的异常);还有就是OOM(OutOfMemoryError:在虚拟机栈是可以动态扩展的情况下,在动态扩展申请内存时,内存不足就会报OOM的异常)
4、通过-Xss来指定虚拟机栈的大小。 -
栈的存储结构和运行原理
虚拟机栈的存储结构时一个个栈帧,每个栈帧对应的一个个方法,如下图所示:
Jvm对栈的操作就只有两个,就是入栈和出栈。在当前线程中,一个时间节点上,只有一个活动的栈帧,我们称做是当前栈帧,对应的方法叫当前方法,对应的类叫当前类。
如果在当前栈帧调用新的方法,就会创建一个新的栈帧入栈,成为当前栈帧;相对的,当前栈帧执行完后出栈,底下一个栈帧就会成为新的栈帧。 -
栈帧的内部结构
1、局部变量表
局部变量表是定义为一个数字数组,主要用于存储方法参数和方法体内的局部变量,存储的数据类型包括基本数据类型、对象引用和返回地址类型;它是建立在线程的栈上,所以不会有线程安全的问题;它所需要的容量大小是在编译期就定下来的,在方法的运行期间是不会改变它的大小的。
我们也可以javap -v 对应的class文件,查看局部变量表对应的信息:
写一个简单的代码:
javap- -v后查看:
2、操作数栈
每一个栈帧里都有操作数栈,他在方法的执行过程中,根据字节码指令,在栈中提取数据去计算,把计算结果压入栈中。它主要用于保存计算过程中的中间结果,同时也是计算过程中变量临时的存储控空间。
每一个栈帧创建的时候,都有对应的操作数栈建立,此时操作数栈的是空的,在方法执行过程中才会有数据进去出。操作数栈的所需深度大小是在编译期就计算好了,存储在code中。
3、动态链接
动态链接,也叫指向运行时常量池的方法引用。
Java编译成字节码文件后,所有的变量和方法都是作为一个符号引用存放到运行时常量池中,当A方法想要调用B方法时,就去运行时常量池找到B方法的符号引用,通过动态链接转换成直接引用实现调用,这就是动态链接的作用。
4、方法返回地址
存放调用该方法的程序计数器的值。
当前方法正常结束后,返回程序计数器的值,好让执行引擎继续往下执行字节码。一个方法的结束分为正常结束和异常结束,正常结束就是返回调用它的方法当时的程序计数器的值,异常退出则要由异常表来确定,异常表不保存在栈帧中。
举个例子:方法A在程序计数器为9时调用了方法B,B就把9存到自己的方法返回地址中,当方法B正常结束后,就返回9,执行引擎根据这个值继续执行字节码。
5、一些附加信息
携带与Java虚拟机实现相关的一些信息,比如对程序调用提供的支持的信息。不是很重要,有些类型的Java虚拟机也可能没有。