赞
踩
下图为JVM内存结构模型:
两种执行方式:
- 解释执行:JVM是由C++语言编写的,其中有C++解释器,负责先将Java语言解释翻译为C++语言。缺点是经过一次JVM翻译,速度慢一点。
- JIT执行:JIT编译器在程序运行时,会将频繁执行的热点代码的字节码编译为本地机器代码,并进行优化,然后把编译后的机器码缓存起来,以备下次使用,从而提高了程序的执行效率。但编译时间较长。
运行时数据区:java虚拟机在执行java程序的过程中会把他所管理的内存划分为若干个不同的数据区域。
直接内存:堆外内存,不是运行时数据区的一部分,但会被频繁使用。没有经过虚拟化。
作用:程序计数器就是行号指示器,是一块较小的内存空间,指向当前线程正在执行的字节码指令的地址。
字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。 分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。
线程是否共享:线程私有。
为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各条线程 之间计数器互不影响,独立存储
如果正在执行的是本地方法,这个计数器值则应为空。
作用:存储当前线程运行Java方法所需的数据、指令、返回地址。
线程是否共享:线程私有。
启动一个线程创建一个虚拟机栈。虚拟机栈是线程私有的,栈的生命周期和线程是一样的。遵循先进后出原则。
栈帧:在每个Java方法被调用的时候,都会创建一个栈帧,并入栈。一旦方法完成相应的调用,则出栈。
栈帧结构:
局部变量表:用于存储方法参数和局部变量的值。存放了编译期可知的基本数据类型、对象引用、returnAddress类型。
局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在栈帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小(局部变量槽(Slot)的数量)。
操作数栈:用于存储计算过程中的操作数和中间结果。
动态链接:将符号引用转换为直接引用的过程。
方法出口:方法出口指向的是调用方法的指令地址,即调用方法的当前执行位置。当一个方法正常结束时,程序计数器(PC)将恢复到该地址,使得程序能够继续执行。
public class Test {
public static void main(String[] args) {
A();
}
private static void A() {
B();
}
private static void B() {
C();
}
private static void C() {
}
}
栈大小:虚拟机栈有大小限制。参数-Xss。默认值取决于平台。
应用场景:系统内存吃紧情况下。
系统剩下100M内存,一个线程堆栈大小默认1M,现在有200个线程分配内存。一般线程请求栈深度不会特别深,这时可以缩小栈大小-Xss256k。
可能产生的异常:
一个方法对应一个栈帧:
public class Student{
public static void main(String[] args) throws Exception{
Student student = new Student();
student.study();
}
public int study() throws Exception{
int a = 1;
int b = 2;
int c = (a + b) * 3;
return c;
}
}
以study()方法为例, 首先使用javap -c
命令查看study()方法的字节码:
最左侧0~12代表字节码地址,字节码偏移量,程序计数器中会记录。程序运行时程序计数器会记录着运行方法字节码的行号。
当执行ireturn
指令时,JVM会将当前方法的返回值(9)推送到调用方法的操作数栈中。
这里以int a = 1;
为例,剩下大家可以结合 详细规范 自己进行分析。
public native int hashCode();
JVM里会有默认参数设置。
通过jps找到运行进程:
然后通过jinfo -flags 3104找到具体参数信息:
然后通过这些信息: JVM申请内存
作用:唯一目的存放Java对象实例。
线程是否共享:线程共享。
所有线程共享的一块区域,在虚拟机启动时创建。
大小:Java堆既可以被实现成固定大小的,也可以是可扩展的,不过当前主流的Java虚拟机都是按照可扩 展来实现的(通过参数-Xmx和-Xms设定)。
特点:
可能产生的异常:
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。