Java&JVM虚拟机模型&类加载机制
一、JVM虚拟机
- JVM虚拟机中的boolean类型
虽然Java虚拟机中定义了boolean类型,但是实际上在Oracle JVM虚拟机中使用的是int类型来编译的。
-
JVM虚拟机模型
如图所示:
编译好的class文件通过类加载将一些信息存入JVM虚拟机中
在运行时会产生多个线程,每一个其中包括三个部分
-
pc regsiter表示注册表,每一个线程都有自己的regsiter,并且在任意时间一个线程都只有一个方法在执行,计数器会存储当前线程正在执行的方法的JVM指令地址
-
jvm-Virtual Machine stack:一个Java虚拟机栈,每一个虚拟机进程只有一个虚拟机栈,并且每一个虚拟机栈存放一个frame,每次方法调用均会创建一个对应的Frame,方法执行完毕或者异常终止,Frame被销毁。一个方法A调用另一个方法B时,A的frame停止,新的frame被创建赋予B,执行完毕后,把计算结果传递给A,A继续执行。一个JVM栈存储本地变量以及执行部分方法的返回值。
其中frame中分为4部分:
1.Local Variables:基本数据类型、引用数据类型、方法返回地址
2.Operand Stack:通过LIFO方式操作数据
3.Dynamic Linking:转换符号引用到直接引用,例如String str=“China”;其中str属于符号引用,"China"在字符串常量池直接引用
4.Method Invocation Completion
-
JVM-heap是虚拟机中的堆,是一个被虚拟机共享的地方,用于存储由类产生的对象和数组区域,在虚拟机启动时就会创建,heap区域会被gc回收,如果heap不足,会抛出OutOfMemoeryError异常
-
JVM-Method Area:方法区,被所有的虚拟机共享,但是不仅仅存放方法,会存放常量池,类的属性,方法数据,方法和构造方法的代码等,在逻辑上也属于heap
-
jvm-Run-Time Constant Pool:JVM 为每个类型都维护了一个常量池,这是一种运行时数据结构,它可以满足一般编程一语言常量表的功能。每个运行时常量池都是从方法区分配的。当 JVM 创建类或接口时,会为其创建它的运行时常量池。
-
JVM-native method stack:这一部分由其他语言实现
-
二、类加载
-
虚拟机把描述类的数据从Class 文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的 Java 类型,这就是虚拟机的类加载机制。
-
java类的生命周期
-
加载:加载字节码文件,会产生一个Class对象,作为访问的入口
类加载会加载3个类库
首先是加载加载 jre/lib/下面的核心类库bootstrap,然后是加载jre/lib/ext下的类库extension,最后是加载用户类 classpath变量定义的目录application
-
链接:确保class文件中的字节流包含的信息符合当前虚拟机的要求。
-
验证文件格式,是否是以魔术开头,验证元数据,看是否存在父类,继承链是否正确,验证字节码、符号引用是否正确
-
在方法区为变量分配内存并设置初始值
-
虚拟机将常量池内的符号引用替换为直接引用的过程。
-
-
初始化:执行类中定义的java代码,此阶段是执行clinit方法的过程。
初始化阶段,才真正开始执行类中定义的 Java 程序代码,此阶段是执行 () 方法的过程。
() 方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并产生的。(不包括构造器中的语句。构造器是初始化对象的,类加载完成后,创建对象时候将调用的 () 方法来初始化对象)
-
使用
-
卸载
-
3.双亲委派机制
自定义的类加载会向父类进行确认,父类则会继续向上一级进行确认,之后由最顶端的bootstrap向下依次传达结果