加载: 通过IO,读取字节码文件;(通过全限定类名获取该类的二进制字节流)
链接:
1)校验: 字节码文件是否符合虚拟机要求
2)准备:为类变量赋初始值
3)解析:将常量池内的符号引用转换为直接引用的过程。
初始化: 就是执行类构造器方法 < clinit > ()的过程,对类的静态变量初始化为指定的值以及执行静态代码块;
1)定义: Java虚拟机对class文件采取的是按需加载方式,也就是需要某类时才会将其加载到内存中生成class对象,而加载某类的class文件时,java虚拟机采用的是双亲委派模式,即把请求交由父类处理,是一种任务委派模式。
2)工作原理图解如下:
如果一个类加载器收到了类加载请求,它并不会第一时间就去加载,而是一级级向上委派;如果父类加载器还有其父类加载器,则进一步向上委派,请求最终将达到顶层的启动类加载器;如果父类加载器可以完成类加载任务,则成功返回,否则再交由子加载器本身去加载,这就是双亲委派模式;
3)优势避免类的重复加载保护程序安全,防止核心API被随意篡改,
4、JVM内存结构例如java.lang包下都是java的核心API,如果我们自定义这样的类则会运行报错:自定义类:java.lang.String 自定义类:java.lang.shkStart
4.1JVM内存结构之------PC寄存器
简单理解:我们每一条指令都有执行顺序,PC寄存器存储着这些指令对应的地址(即将要执行的指令代码),由执行引擎读取下一条指令。
它是程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都需要它来完成,是唯一一个在Java虚拟机规范中没有规定任何Out of Memory Error情况的区域;
两个常见问题:
1)使用PC寄存器存储字节指令地址有什么用?(为什么用PC寄存器记录当前线程的执行地址呢?)
因为CPU需要不停的切换各个线程,这时候切换回来之后,需要知道从哪开始继续执行。
2)PC寄存器为什么被设定为线程私有?
为了能够准确的记录各个线程正在执行的当前字节码指令地址,最好的办法自然是为每一个线程都分配一个PC寄存器。



