**
第一篇学习博客,学习JVM调优总结
**
前言
第一点和第二点是之前没掌握的,所以特意提出来,第三点的脑图比较好理解
一、类加载运行全过程
其中loadClass的类加载过程有如下几步:
加载 >> 验证 >> 准备 >> 解析 >> 初始化 >> 使用 >> 卸载
-
加载:在硬盘上查找并通过IO读入字节码文件,使用到类时才会加载,例如调用类的
main()方法,new对象等等,在加载阶段会在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口 。(简单就是:在硬盘里面的class类文件,变成二进制加载到java虚拟机里面去) -
验证:校验字节码文件的正确性(个人理解:最起码字节码是以“CA FE BA BE”开头,才是标准的字节码文件)
-
准备:给类的静态变量分配内存,并赋予默认值 (比如int赋值为0默认值,java虚拟机规定的)
-
解析:将符号引用替换为直接引用,该阶段会把一些静态方法(符号引用,比如main()方法)替换为指向数据所存内存的指针或句柄等(直接引用),这是所谓的静态链接过程(类加载期间完成),动态链接是在程序运行期间完成的将符号引用替换为直接引用。
A.符号引用
CONSTANT_Class_info
constant_field_info
Constant_method_info等类型的常量
它与虚拟机实现的布局无关,引用的目标并不一定要已经加载到内存中。各种虚拟机实现内部布局可以各不相同,但是它们能接受的符号引用必须是一致的,因为符号引用的字面量形式明确定义在java虚拟机规范的class文件格式中
B.直接引用
1.可以是指指向目标的指针
2.相对偏移量
3.一个能间接定位到目标的句柄。
如果有了直接引用,那引用的目标必定已经在内存中存在
- 初始化:对类的静态变量初始化为指定的值,执行静态代码块
类被加载到方法区中后主要包含 运行时常量池、类型信息、字段信息、方法信息、类加载器的 引用、对应class实例的引用等信息。
类加载器的引用:这个类到类加载器实例的引用
对应class实例的引用:类加载器在加载类信息放到方法区中后,会创建一个对应的Class 类型的 对象实例放到堆(Heap)中, 作为开发人员访问方法区中类定义的入口和切入点。
- 注意,主类在运行过程中如果使用到其它类,会逐步加载这些类。 jar包或war包里的类不是一次性全部加载的,是使用到时才加载。 示例:pandas 是基于NumPy 的一种工具,该工具是为了解决数据分析任务而创建的。
二、jvm结构图
1、右上角的是类加载系统,也就是 加载-》验证-》准备-》解析-》初始化
“运行时数据区”
2.1、共享的资源
2.2.1、堆(java heap):
a.所有线程共享的
b.所有对象实例都在这里分配内存。
c.堆有分代机制
2.2.2 方法区
a.保存装载的累的元信息:类型的常量池,字段,方法信息,方法字节码;JDK6时,String等敞亮信息置于方法区,jdk7移到了堆中
b.通常和永久区(Perm关联在一起)
c.永久代在JDK1.8时候变成了元空间,直接存在直接内存里面
2.2 私有的资源
2.2.1、栈:
a.线程私有
b.栈有一系列帧组成,所以也叫帧栈
c.帧保存一个方法的局部变量(局部变量表)、操作数栈、常量池指针
2.2.2、本地方法栈
2.2.3、PC寄存器
a.每个线程启动的时候,都会创建一个PC(Program Counter,程序计数器)寄存器。
b.PC寄存器里保存有当前正在执行的JVM指令的地址。
c.每一个线程都有它自己的PC寄存器,也是该线程启动时创建的。
d.它是一块很小的内存空间,几乎可以忽略不计
e.线程私有,生命周期保持一致,唯一一个没有OOM的区域,没垃圾回收
f.并行的时候,CPU会来回切换线程执行,切换回来之后,就知道下一步做什么了