java 每当启用一个线程时,JVM就为他分配一个Java栈。这个栈是线程私有的,其生命周期与线程一样。其他的线程是不可能访问到这个线程栈上的数据,所以栈里的数据是不存在同步问题的
Java栈是以栈帧为单位来保存其运行状态的,当线程在执行某个方法时,就会创建一个栈帧并压入栈中。
栈帧包括局部变量表,操作数栈,动态链接,返回值等信息。
局部变量表:是在编译期间就完成的分配,所以一个方法需要在栈帧中分配多大的局部变量空间是确定的,在运行期间这个空间大小是不会改变的。它存放了编译器可知的各种基本数据类型,以及对象的引用指针和returnAddress(指向一条字节码指令的地址),除了long,double需要占用两个slot(局部变量空间)其余只占用一个,举例
public static int runClassMethod(int i,long l,float f,double d,Object o,byte b) {
return 0;
}
public int runInstanceMethod(char c,double d,short s,boolean b) {
return 0;
}
runInstanceMethod的局部变量区第一项是个reference(引用),它指定的就是对象本身的引用,也就是我们常用的this,但是在runClassMethod方法中,没这个引用,那是因为runClassMethod是个静态方法
操作数栈:一个临时数据区。比如在A方法中调用B方法,在调用B方法时会创建一个栈帧压入该线程的栈顶,执行完B方法后,该栈帧出栈。将B的返回结果值存入操作数栈中,供A继续执行。
动态链接:是指对调用方法的引用,以操作数栈的例子来说,动态链接就是A对B的引用
返回值:方法的返回分为正常结束和异常终止。如果是通过return正常结束,则当前栈帧从Java栈中弹出,恢复发起调用的方法的栈。如果方法又返回值,JVM会把返回值压入到发起调用方法的操作数栈。如果是异常终止,栈帧调用此方法异常引用表(栈帧中保存着一个对此方法异常引用表的引用)。当异常抛出时,JVM给catch块中的代码。如果没发现,方法立即终止,然后发起调用方法的上下文抛出异常。