不破不立,人类最宝贝的品质就是勇敢和过去告别
—— 24.6.21
一、类的加载时机
1.new对象
2.new子类对象(new子类对象先初始化父类)
3.执行main方法
4.调用静态成员
5.反射,创建Class对象
这五种情况就可以让类加载到内存
类加载过程
1.问题:谁将class文件加载到了内存?
jvm
2.问题:jvm利用了什么技术将class文件加载到了内存?
IO流技术
3.问题:jvm相当于一个大"boss",所以有很多事情不是jvm直接插手,比如加载class文件就不是ivm直接插手干的,那么是jvm下面的谁干的呢?
"类加载器"->ClassLoader
二、类加载器 ClassLoader
1.概述
在jvm中,负责将本地上的class文件加载到内存的对象_ClassLoader
2.分类
① BootstrapClassLoader:根类加载器->C语言写的,我们是获取不到的也称之为引导类加载器,负责java的核心类加载的
比如:System,String等
jre/lib/rt.jar下的类都是核心类
② ExtClassLoader:扩展类加载器
负责jre的扩展目录中的jar包的加载
在idk中ire的lib目录下的ext目录
③ AppClassLoader:系统类加载器
负责在jvm启动时加载来自java命令的c1ass文件(自定义类),以及classPath环境变量所指定的jar包(第三方jar包)
不同的类加载器负责加载不同的类
3.三者的关系(从类加载机制层面):
AppclassLoader的父类加载器是ExtclassLoader
ExtClassLoader的父类加载器是BootstrapclassLoader
但是:他们从代码级别上来看,没有子父类继承关系->他们都有一个共同的父类->ClassLoader
4.获取类加载器对象:getClassLoader()是class对象中的方法
类名.class.getClassLoader()
5.获取类加载器对象对应的父类加载器
ClassLoader类中的方法:ClassLoader
getParent —> 没啥用
6.双亲委派(全盘负责委托机制)
a.Person类中有一个string
Person本身是AppclassLoader加载
String是BootstrapClassLoader加载
b.加载顺序:
Person本身是App加载,按道理来说String也是App加载
但是App加载string的时候,先问一问Ext,说:Ext你加载这个string吗?
Ext说:我不加载,我负责加载的是扩展类,但是app你别着急,我问问我爹去->boot
Ext说:boot,你加载String吗?
boot说:正好我加载核心类,我加载
7.类加载器的cache(缓存)机制(扩展):
一个类加载到内存之后,缓存中也会保存一份儿,后面如果再使用此类,如果缓存中保存了这个类,就直接返回他,如果没有才加载这个类,下一次如果有其他类在使用的时候就不会重新加载了,直接去缓存中拿,保证了类在内存中的唯一性
向上委托,向下查找
类加载器的双亲委派和缓存机制共同造就了加载类的特点:保证了类在内存中的唯一性