jvm分为三个主要的子系统:
(1)类加载系统 (2)运行时数据区 (3)执行引擎
数据加载进运行时数据区,执行引擎从数据区获取执行
一、类加载系统
处理动态类加载功能,当运行时首次引用一个类时,它加载、链接、初始化该类文件
1.加载
启动类加载器(BootStarp class Loader)、扩展类加载器(Extension class Loader)、应用程序类加载器(Application class Loader)三个组件完成此操作:
- 启动类加载器- 父 -从启动类路径中加载类(rt.jar,就是java基本类,javaDoc中的所有),拥有最高优先级
-
扩展类加载器- 子 -负责加载
ext
目录的类(jre\lib
) - 应用程序类加载器- 孙 -加载应用程序级别类路径。
加载机制:
- 全盘负责:类加载器加载一个类时,默认该类依赖和引用的其它类也由该加载器负责.
- 双亲委派: 先让父类加载试图加载的Class依次递归,父类加载不了才自己加载。
- 缓存机制:所有加载过的类都会被缓存,程序使用类时加载器先从缓存中寻找。所以修改Class后必须重启JVM。
用户类加载器=>应用程序类加载器=>扩展类加载器=>启动类加载器
双亲委派优点:1.防止重复加载 2.防止底层传入虚假的核心JavaAPI
2.链接
- 校验-字节码校验器会校验生成的字节码是否正确,失败提示校验错误信息。
- 准备-分配内存并初始化默认值给所有静态变量。
-
解析-所有的符号内存引用被方法区(
Method Area
)的原始引用替代。
3.初始化
这是类加载的最后阶段,静态变量赋值,静态代码块被执行。
二、运行时数据区
之前的文章jmm(java内存模型)有讲到,不再赘述。
三、执行引擎
将数据区内的字节码逐段执行。
1.解释器
能快速解析字节码,但是对于重复的方法会每次都重新解析
2.即时编译器JIT(Just In Time Compiler)
为了解决解释器的缺点,当被判定为重复代码将使用JIT编译,并重复利用。
其它:
- 中间代码生成器-生成中加代码
- 代码优化器 -负责优化上面生成的代码
- 目标代码生成器-生成既期待吗或本机代码
- 探测器(Profiler)-负责寻找被多次调用的方法。
垃圾回收机制(GC)
之后另起一章讲述
JVM 参数
- 堆:
-Xms
-Xmx
(一般设置一样的值,否则内存抖动) - 线程:
-Xss
线程堆栈调用深度 - 新生代初始大小:
–XX:NewSize–XX:MaxNewSize
指定新生代初始大小和最大大小。 -
-XX:NewRatio
是年老代 新生代相对的比例 -
-XX:SurvivorRatio
配置的是在新生代里面Eden
和一个Servive
的比例