3.5 KiB
3.5 KiB
JVM简介
优势:
- write once run everywhere
- 内存管理
jvm空间加载运行
参考jvm规范:https://docs.oracle.com/javase/specs/index.html
加载
- 类加载器
- Bootstrap:加载环境变量中$JAVA_HOME/jre/lib/rt.jar里所有的class,不是ClassLoader子类。
- Extension:加载java平台中扩展功能的一些jar包。包括环境变量中$JAVA_HOME/jre/lib/*.jar或-Djava.ext.dirs指定目录下的jar包。
- Application:加载classpath中指定的jar包及目录中class。
- Custom自定义:应用程序根据自身需要自定义的ClassLoader,如tomcat、jboss都会根据j2ee规范自行是实现ClassLoader
- 双亲委派
- 写类String写main方法无法启动,Bootstrap Class Loader 中已经加载了String,其中无main方法,无法启动。委派给父类,所有父类都加载不到才会有子类加载。不让jdk代码被修改。
链接
- 验证:字节码校验器会检查生成的字节码是否正确,如果验证失败会停止运行
- 准备:对于所有静态变量内存分配和默认值分配
- 识别:解析或识别是从运行时常量池的符号引用中动态具体的过程。
初始化
- 类或接口的初始化由执行类或接口初始化方法构成,这里所有的静态变量与原来的值将被指派,静态块将被执行。
线程空间
- 线程共享
- Heap堆:在虚拟机启动时创建,此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。存储数据不是线程安全的,堆和常量池空间不足时会触发OutOfMemoryError
- MetaSpace元数据区:在1.8之后取代Method Area 方法区永久代,不在jvm中,使用本地内存,受操作系统内存限制。存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
- 线程隔离
- VM Stack虚拟机栈:每个线程创建一个单独的运行时堆栈。对于每一个方法调用,一个称为栈内存栈帧被创建。所有局部变量将被创建在栈内存中。栈区域时线程安全的,因为他不是一个共享资源。
- Navive Method Stack本地方法栈:与虚拟机所发挥的作用是非常相似的,其区别不过是虚拟机栈为虚拟机执行java方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的Native方法服务。
- Program Counter Register 程序计数器:每个线程必须分开程序计数器登记,当前执行的指令一旦被执行,程序计数器更新下一个指令。一块较小的内存空间。他的作用可以看做是当前线程所执行的字节码的行号指示器。
运行时数据区
-
程序计数器:指向当前线程正在执行的字节码指令的地址和行号
-
虚拟机栈:存储当前线程运行方法所需的数据、指令、返回地址(动态链接,注入使用查找实例)。
-
- 栈帧:局部变量表,操作数栈、动态链接、出口
- javap -c -v .class > p.txt
-
本地方法 native
-
方法区 类信息、常量、静态变量、JIT(just in time)编译时信息(永久代对方法的实现)
-
Heap 堆(内存模型)存放new出来的对象
-
- 新生代
- 老年代
- 永久代/元数据区(方法区、非堆)