简单介绍类加载过程:
JVM的内存结构:
不可共享:
虚拟机栈
本地方法栈
程序计数器
可共享:
堆
方法区
1. 类加载器的基本使用:加载,验证,准备,解析,初始化。
类加载器具有三个,引导类加载器,扩展类加载器,系统类加载器。
类加载器系统负责从文件系统或者网络中加载Class文件的加载,class文件在文件开头有特定的文件标识。
ClassLoader 只负责加载class文件,至于是否可以运行由 execution Engine 决定
加载后的类信息存放在 方法区中,常量池也存放于方法区中。
1.2 详细的类加载过程
2. 双亲委派机制
String类的双亲委派机制,我们通过手写的一个String类的一个静态代码块打印,并通过另一个StringTest类去new一个String对象后,我们发现String的静态代码块没有被执行,因此执行的是Java核心类库中的String类。
2.1 双亲委派机制的第一种模型
双亲委派机制的原理是:
当一个类被加载进去之后交给类加载器,类加载器不会再第一时间进行加载,首先会寻找它的父类加载器,若还有父类则继续向上委托,把这个请求委托给父类的加载器去执行。
若是父类加载器可以完成类加载任务,就成功返回,倘若是父类加载器无法完成此加载任务,自加载器才会自己进行加载,这就是双亲委派机制。
public class Diparents { public static void main(String[] args) { String s =new String(); System.out.println("hello,atguigu.com"); } }
package java.llang; public class String { static{ System.out.println("我是自定义的String类的静态代码块"); } }
直接向上进行委托,到引导类加载器,引导类加载器执行java核心库中的String.
系统类加载器:(AppClassLoader)
java 语言编写,派生与ClassLoader类
它负责加载环境变量classpath或者系统属性 java.class.path 指定路径下的类库。
通过ClassLoader # getSystemClassLoader()方法可以获得该类加载器。
扩展类加载器的加载路径:
派生于ClassLoader类
从java.ext.dirs系统属性指定的目录中进行加载或者从jre/lib/ext子目录下加载类库。
2.2 双亲委派机制的第二种模型】
2.3 双亲委派机制的优势是什么:
(1)防止类被重复加载,因为类加载器本身具有层次性,因此不可能存在多次加载。
(2)保护程序的安全性,防止核心API被随意的篡改。
自定义类:java.lang.String 不能拥有main方法,因此不能执行。
自定义类:java.lang.Sasd; 被引导类加载器进行加载后,不能执行。
2.4 什么是沙箱安全机制:
我们在使用引导类加载器对String类进行加载的时候,首先会找到JDK自带的文件(rt.jar)包中java\lang\String.class,报错信息会说没有main方法,就是因为加载的是rt.jar包中的类。可以对源码进行保护、因此叫沙箱安全机制。
2.5 两个类完全一样是怎么定义的
在jvm中定义的两个类完全一样:
1. 所在的包名是完全一样的
2. 所在的包中我们加载所用的类加载器是一样的,称之为两个类是完全一样的。
2.6 类的主动使用与被动使用
3. 运行时数据区:
3.1 PC寄存器
1. 使用PC寄存器存储字节码指令地址有什么用?为什么使用PC寄存器记录当前线程的执行地址呢?
CPU需要不断的切换线程,因此需要PC寄存器来记录到底到底执行到哪里了。
2. PC寄存器为什么设定为线程私有?
为了能够准确的记录当前字节码指令地址,最好的办法是为每一个线程都分配一个PC寄存器。防止之间相互干扰。
将指令地址(偏移地址) 存放在PC寄存器里面,执行引擎开始执行相应的操作。
作用:存放下一个指令的地址,有执行引擎读取下一条指令。
什么是并行跟串行的区别:就是垃圾回收线程跟用户线程不能并发的去执行。垃圾回收线程可以有很多个同时执行,但是此时就不能执行用户线程。
3.2 虚拟机栈
栈跟堆的区别是,栈是来解决程序运行问题,例如它应该怎么执行,如何处理数据,
堆是用来存储问题,数据怎么放,放在哪里。
3.1 一个栈帧代表一个方法。
一次方法的调用伴随着栈的使用,声明周期是与线程是相同的。
作用:
保存方法的局部变量,部分结果,并参与方法的调用和返回。
栈不存在垃圾回收但是会存在OOM
3.2 java虚拟机栈中可能出现的异常
1. 可能会出现StackOverflowError异常,我们在java虚拟机中,如果线程请求分配的栈容量超过java虚拟机栈的最大容量后就会出现此异常
2. 如果java虚拟机栈可以动态扩展,并且在尝试扩展后无法申请到足够的内存,或者在创建新的线程时候没有足够的内存去创建对应的虚拟机栈。因此java虚拟机会抛出一个OutOfMemoryError异常。
3.3 如何设置栈内存的大小
我们可以设定参数-Xss 选项来设定线程的最大栈空间,栈的大小决定了函数调用的最大深度。
栈内部的结构:
栈中到底存储什么:
每个线程都有自己的栈,栈的数据存储是以栈帧的方式进行。
在这个线程上每个方法都有自己的栈帧,
栈帧是一个内存区块,是一个数据集,存放着方法执行过程中的各种数据信息。
main 主线程对应栈帧1
局部变量表(本地变量表):
1 局部变量数组,定义为一个数字数组,主要存储方法的形参,以及定义在方法体内的局部变量。这些数据类型包含了各类的基本数据类型,以及对象引用,以及returnAddress类型。
2 由于局部变量表是建立在线程的栈上,是线程的私有数据,因此不存在数据安全问题。
3 局部变量表所需的容量大小是在编译期确定下来的。并保存在方法的Code属性的maximum local variable数据项中。在方法运行期间是不会改变局部变量表的大小的。