ClassLoader类加载器顺序Demo测试与双亲委派源码解读

上一篇 <<<javap命令反查汇编指令汇总
下一篇 >>>自定义SPI和热部署技术破坏类加载器的双亲委派模式


java是逻辑程序,class是虚拟机指令程序。
类加载器:将我们class文件读取到内存中。

class文件的来源

  1. 自己写的java源代码 编译成class文件 硬盘读取
  2. 通过网络的方式下载class文件
  3. War、Jar 解压之后都是class文件
  4. 从数据库中读取class文件
  5. Java动态代理模式 反射/cglib 生成代理class文件

类加载过程

1、加载(Loading)

a、class字节码文件数据转换成方法区中的运行时数据结构【字节码文件可以是本地,也可以来自于网络、光盘、硬盘等二进制】
b、在堆中创建代表类的class对象(注意不是目标类对象,用newInstance就可以生成对象),作为方法区类数据的访问入口,该对象封装了类在方法区中的数据结构,并且向用户提供了访问方法区数据结构的接口,即Java反射的接口。

2、链接(Linking)

1)、验证(Verification):确保加载的类信息符合JVM规范,没有安全方面的问题【JVM校验】
a.文件格式验证:主要验证字节流是否符合Class文件格式规范,并且能被当前的虚拟机加载处理。例如:主,次版本号是否在当前虚拟机处理的范围之内。常量池中是否有不被支持的常量类型。指向常量的中的索引值是否存在不存在的常量或不符合类型的常量。
b.元数据验证:对字节码描述的信息进行语义的分析,分析是否符合java的语言语法的规范。
c.字节码验证:最重要的验证环节,分析数据流和控制,确定语义是合法的,符合逻辑的。主要的针对元数据验证后对方法体的验证。保证类方法在运行时不会有危害出现。
d.符号引用验证:主要是针对符号引用转换为直接引用的时候,是会延伸到第三解析阶段,主要去确定访问类型等涉及到引用的情况,主要是要保证引用一定会被访问到,不会出现类等无法访问的问题。

2)、准备(Preparation):正式为类变量(static变量)分配内存并设置类变量初始值的阶段,这些内存都将在方法区中进行分配【空间分配】
3)、解析(Resolution):虚拟机常量池的符号引用替换为字节引用过程【静态和常量池引用替换】

3、初始化(Initialization)

初始化阶段是执行类构造器<clinit>()方法的过程。

  • 在多线程环境中会被正确加锁和同步
  • 自动收集类中变量的复制动作,以及static块的语句合并产生,static从上往下执行,如果父类未初始化,则先初始化父类。

双亲委派层次结构

a、启动(Bootstrap)类加载器【加载JVM自身的类,由C++实现,没有父类,文件名写死】
Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类,位于<JAVA_HOME>/lib路径下的核心类库和-Xbootclasspath参数指定的路径下的jar包

/**
 * 启动类加载器的职责(注意最后的classes目录没有的话自己创建)
 * /Home/jre/lib/resources.jar
 * /Home/jre/lib/rt.jar
 * /Home/jre/lib/sunrsasign.jar
 * /Home/jre/lib/jsse.jar
 * /Home/jre/lib/jce.jar
 * /Home/jre/lib/charsets.jar
 * /Home/jre/lib/jfr.jar
 * /Home/jre/classes
 */
public static void bootstrapClassLoader() {
    String property = System.getProperty("sun.boot.class.path");
    List<String> list = Arrays.asList(property.split(";"));
    list.forEach((t) -> {
        System.out.println("启动类加载器目录:" + t);
    });
}

b、扩展(ExtClassLoader)类加载器【由Java语言实现,父类加载器为null】
由Java语言实现的,是Launcher的静态内部类,它负责加载<JAVA_HOME>/lib/ext目录下或者由系统变量-Djava.ext.dir指定位路径中的类库,开发者可以直接使用标准扩展类加载器。

/**
 * 扩展类加载器
 * /Home/jre/lib/ext
 * /Library/Java/Extensions
 * /Network/Library/Java/Extensions
 * /System/Library/Java/Extensions
 * /usr/lib/java
 */
public static void extClassLoader() {
    String property = System.getProperty("java.ext.dirs");
    List<String> list = Arrays.asList(property.split(";"));
    list.forEach((t) -> {
        System.out.println("扩展类加载器" + t);
    });
}

c、系统 (AppClassLoader) 类加载器【由Java语言实现,父类加载器为ExtClassLoader】
也称应用程序加载器是指,它负责加载classpath路径,该类加载是程序中默认的类加载器,通过ClassLoader#getSystemClassLoader()方法可以获取到该类加载器。

/**
 * app 类加载器(除了包含启动类和扩展类,还包括自己写的代码)
 * /target/classes
 */
public static void appClassLoader() {
    String property = System.getProperty("java.class.path");
    List<String> list = Arrays.asList(property.split(";"));
    list.forEach((t) -> {
        System.out.println("应用类加载器" + t);
    });
}

d、自定义类加载器,父类加载器肯定为AppClassLoader。

双亲委派模式原理及好处

原理:
如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行(存在递归),倘若父类加载器无法完成此加载任务,子加载器才会尝试自己去加载,这就是双亲委派模式。

好处:
a、避免类的重复加载,当父亲已经加载了该类时,就没有必要子ClassLoader再加载一次。
b、可以防止核心API库被随意篡改,比如java.lang是核心API包,扩展时强制加载将会报出如下异常:java.lang.SecurityException: Prohibited package name: java.lang

类加载器的触发节点

1.调用类的静态方法
2.invokeStatic 调用静态方法
3.Main
4.New
5.Class.formname
6.子类初始化一定会初始化父类
初始化一个类,一定会触发类加载器
类加载器加载了该类,但该类不一定完成初始化。

类加载器效果测试

// 当前classLoader【正常】:sun.misc.Launcher$AppClassLoader@18b4aac2
System.out.println(JaryeEntity.class.getClassLoader());
// 扩展classloader:sun.misc.Launcher$ExtClassLoader@5f8ed237
System.out.println(JaryeEntity.class.getClassLoader().getParent());
// 启动classLoader:null(因为底层是C语言写的)
System.out.println(JaryeEntity.class.getClassLoader().getParent().getParent());
// 把class文件放到启动类的/Home/jre/classes目录下,则打印结果为null
System.out.println(JaryeEntity.class.getClassLoader());
// Object在lang包下,属于启动类加载,结果为null
Object o = new Object();
System.out.println(o.getClass().getClassLoader());

双亲委派源码解读

ClassLoader.getSystemClassLoader().loadClass("com.mysql.jdbc.Driver")

类加载器核心方法

a、loadClass(String)
ClassLoader类自己实现的,不再建议用户重写但用户可以直接调用该方法
加载时从缓存中获取,没有则委派给启动类加载,如果仍然未找到,则调用findClass()方法去加载。
运行时加载自己指定的类,那么我们可以直接使用this.getClass().getClassLoder.loadClass("className"),这样就可以直接调用ClassLoader的loadClass方法获取到class对象。

b、findClass(String)
JDK1.2之后建议把自定义的类加载逻辑写在findClass()方法中,在loadClass()方法中被调用的,这样就可以保证自定义的类加载器符合双亲委托模式
ClassLoader类中并没有实现findClass()方法的具体代码逻辑,取而代之的是抛出ClassNotFoundException异常
findClass方法通常是和defineClass方法一起使用的

c、defineClass(byte[] b, int off, int len)
defineClass()方法是用来将byte字节流解析成JVM能够识别的Class对象(ClassLoader中已实现该方法逻辑)
在自定义类加载器时,会直接覆盖ClassLoader的findClass()方法并编写加载规则,取得要加载类的字节码后转换成流,然后调用defineClass()方法生成类的Class对象

d、resolveClass(Class≺?≻ c)
使用该方法可以使用类的Class对象创建完成也同时被解析。
前面我们说链接阶段主要是对字节码进行验证,为类变量分配内存并设置初始值同时将字节码文件中的符号引用转换为直接引用。


相关文章链接:
<<<JVM整体内存结构的图解,直观明了
<<<javap命令查看对象信息及操作方法在JVM层的实现原理
<<<javap命令反查汇编指令汇总
<<<自定义SPI和热部署技术破坏类加载器的双亲委派模式
<<<JVM中对象如何完成空间分配和初始化工作
<<<JVM元空间(方法区)和栈内存溢出原因及解决方案
<<<JVM堆内存溢出和内存泄露问题定位和解决
<<<JVM常见死锁问题产生原因和多种诊断方式
<<<服务器CPU飙升为100%问题排查及如何避免
<<<JVM内存诊断命令和排查工具汇总
<<<JVM新生代老年代算法汇总图解
<<<JVM垃圾回收不要手动System.gc的真正原因
<<<JVM垃圾回收引用计数法和根搜索算法图解
<<<JVM垃圾回收STW(Stop-The-World)代码演示
<<<JVM垃圾回收器的发展历程及使用场景汇总
<<<JVM串行并行垃圾回收器的关注点
<<<一张图看懂CMS垃圾回收器的底层原理
<<<G1能作为JDK9默认垃圾回收器的优势分析
<<<CMS和G1的漏标问题解决及三色标记算法图解
<<<GC中新生代进入老年代的方式汇总
<<<GC常用日志参数配置及分析工具说明
<<<FullGC、MinorGC、STW等常见问题如何解答
<<<JVM性能调优的评估指标及调优示例

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容