一:类从加载到使用的过程
加载-->验证--> 准备-->解析-->初始化-->使用-->卸载
1):加载
当类需要使用的时候,就会从对应的.class文件加载到内存中去
2):验证
验证阶段就是根据java虚拟机的规范,来校验你加载的.class文件是否符合指定的规范,然后再交给jvm来执行
3):准备
准备阶段就是给加载的类分配一定的内存空间,然后给类内部的成员变量分配内存空间,并分配给默认值
4):解析
解析阶段就是把符号引用变为直接引用
5):初始化
当类被实例化对象(new Obj())的时候,此时就会触发类的加载到初始化的全过程,把这个类准备好,然后再实例化一个对象出来,或者是包含 main()方法的主类,必须是马上初始化的
此外还有一个非常重要的准备就是初始化一个类的时候,发现父类还没有初始化,那么必须先初始化它的父类,
注意这里和接口的初始化有点区别,,一个接口在初始化时,并不要求其父接口全部都完成了初始化,只要在真正使用到父接口的时候(如引用接口中定义的常量)才会初始化。
初始化
二:类加载器
1):启动类加载器
Bootstrap ClassLoader 主要负责加载我们机器上安装的java 目录下的核心类的,一旦java启用就会依托启动类加载来加载你java安装目录下的”lib“目录下的核心类库
2)拓展类加载器
Extension ClassLoader 主要负责加载你java目录下的”lib/ext“目录下的类
3)应用程序类加载器
Application ClassLoader 这个类加载器就是加载"ClassPath"环境编辑所指定的路径下的类,可以理解为就是加载你java代码
4)自定义类加载器
处理上面几种,还可以自定义类加载器,去根据自己的需求去加载自己的类
5)双亲委派机制
jvm的类的加载器是有亲子层结构的,就是说启用类加载器是最上层的,拓展类加载器在第二层,应用程序类加载器在第三层,最后一层是自定义类加载器;
注意:
三:JVM的内存区域
1)存放类的方法区
方法区在java8之前是代表JVM中的内存区域,主要是用了存放从”.class“文件加载进来的类,还会有一个些类似常量池的的东西放到该区域,但是在java8之后,这块区域改为“Meta Space”元数据区,
2)执行代码指令用的程序计数器
java代码会被编译成字节码。对应各种字节码指令,所以当JVM加载类信息到内存之后,实际上就会使用自己的字节码执行引擎,去执行我们写的代码,如图:
那么,当执行字节码指令的时候,JVM就需要一个特殊的内存区域,那就是程序计数器,这个程序计数器就是用来存放当前执行字节码指令的位置,也就是记录目前执行到哪一条字节码指令
3) java虚拟机栈
java虚拟机栈就是用来存放每个方法内存的局部遍历的数据,每个线程都有自己的java虚拟机栈,如果一个线程执行了一个方法,就会对这个方法调用创建一个栈针,栈针里就有这个方法的局部遍历表,操作数栈,动态链接,方法出口等,
JVM中的java虚拟机栈的作用就是调用任何方法时,都会给方法创建栈针,在栈针里存放这个方法对应的局部变量之类的数据,方法执行完毕后就出栈,
4)JAVA堆内存
java堆内存就是用来存放我们在代码中创建的各种对象,
5)本地方法栈
在调用native方法的时候就会有线程对应的本地方法栈,这和java虚拟机栈类似,存放着各种native方法的局部变量表之类的信息
6)其他内存区域
还有一个区域不属于jvm,可以调用Nio中的allocateDirect这种api可以在java堆外内存分配空间,然后通过java虚拟机的DirectByteBuffer来操作和引用堆外内存空间
7)核心区域串讲
1:当JVM启用,会将kafka类加载到内存里,然后有一个main线程,开始执行kafka中的main方法,main线程关联了一个程序计数器,记录main线程执行的指令
2:其次就是在main线程在执行main方法时,会在main线程关联的java虚拟机栈里,压入一个main方法的栈针,接着发现需要创建一个ReplicaManager 类的实例对象,此刻就会把ReplicaManager类加载到内存中,然后回去创建一个ReplicaManager的对象分配在Java堆内存中,并且在main()方法的栈针里的局部遍历表中引入一个“replicaManager”变量,让他引用ReplicaManager 对象在Java堆内存中的地址,
3:接着main线程开始执行ReplicaManager 中的方法,一次把自己执行到的方法对应的栈针压入到自己的java虚拟机栈,执行完方法后在把对应的栈针从自己的java虚拟机栈里出栈,