类加载机制----------------------------------------形成JVM可以直接使用的Java类型的过程
重要判断Class是否相等(类加载器必须是同一个),要看2个准则
Class clazz1; Class clazz2;
(clazz1==clazz2&&clazz1.getClassLoader()==clazz2.getClassLoader())
JVM把class文件加载到内存,并对数据进行校验、准备、解析、初始化,最终形成JVM可以直接使用的Java类型的过程。
加载+连接(验证、准备、解析)+初始化
1、类加载的唯一产物就是:堆中类Class对象(方法区类数据访问入口)
将class字节码文件加载到内存中,并将这些数据转换成方法区中的运行时数据(静态变量、静态代码块、常量池等),
在堆中生成一个Class类对象代表这个类(反射原理),作为方法区类数据的访问入口。
类所属的放到方法区(模板数据)
对象数据放到堆中
2、链接
将Java类的二进制代码合并到JVM的运行状态之中。
• 验证
确保加载的类信息符合JVM规范,没有安全方面的问题。
• 准备
正式为类变量(static变量)分配内存并设置类变量初始值的阶段,
这些内存都将在方法区中进行分配。注意此时的设置初始值为默认值,具体赋值在初始化阶段完成。
• 解析
虚拟机常量池内的符号引用替换为直接引用(地址引用)的过程。
3、初始化(具体赋值阶段) 执行类的<clinit>()方法
初始化阶段是执行类构造器<clinit>()方法的过程。
类构造器<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static块)中的语句合并产生的。
当初始化一个类的时候,如果发现其父类还没有进行过初始化、则需要先初始化其父类。
虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确加锁和同步。
5种主动加载
1、访问静态变量(不是常量和已知量)+静态方法
2、new 对象
3、子类初始化,父类
4、Class.forName()
5、启动类
类加载器
BootStrap jre/lib/rt.jar
ExtClassLoader jre/ext/*
AppClassLoader classpath
CommonClassLoader common
CatalinaClassLoader SharedClassLoader
WebAppClassLoader WebAppClassLoader
WebAppClassLoader:loadClass
按照顺序加载,ignore ClassNotFoundException异常
(0) Check our previously loaded local class cache
(0.1) Check our previously loaded class cache
//对于一些未加载的非基础类(Object,String等)还是由System.class加载,如果加载不到,忽略异常,继续向下
(0.2) Try loading the class with the system class loader, to prevent the webapp from overriding J2SE classes
(0.5) Permission to access this class when using a SecurityManager
// 委托加载开关是否已经打开,如果打开走委托机制
(1) Delegate to our parent if requested
(2) Search local repositories
(3) Delegate to parent unconditionally
JasperLoader:出现的目的就是为了被丢弃(热加载、热修复原理)
JasperLoader的加载范围仅仅是这个JSP文件所编译出来的那一个.Class文件,
它出现的目的就是为了被丢弃:当Web容器检测到JSP文件被修改时,会替换掉目前的JasperLoader的实例,并通过再创建一个新的Jsp类加载器来实现JSP文件的HotSwap功能。