学习笔记:深入分析ClassLoader工作机制

深入分析ClassLoader工作机制

ClassLoader除了能将Class加载到JVM中之外,还有一个重要的作用就是审查每个类应该由谁加载,它是一种父优先的等级加载机制。

Classloader类结构分析

我们常会用到或扩展ClassLoader,主要会用到如下几个方法:

ClassLoader
defineClass(byte[],int,int)
findClass(String)
leadClass(String)
resolveClass(Class<?>)

其中defineClass方法用来将byte字节流解析成JVM能够是别的Class对象,有了这个方法意味着我们不仅仅可以通过class文件实例化对象,还可以通过其他方式实例化对象,如我们通过网络接收到一个类的字节码,拿这个字节码流直接创建类的实例化对象。注意,如果直接调用这个方法生成类的Class对象,这个类的Class对象还没有resolve,这个resolve将会在这个对象真正实例化时才进行。

defineClass通常是和findClass方法一起使用的,我们通过直接覆盖ClassLoader父类的findClass方法来实现类的加载规则,从而取得要加载类的字节码。然后调用defineClass方法生成类的Class对象,如果你想在类被加载到JVM中时就被链接(Link),那么可以接着调用另外一个resolveClass方法,当然你也可以选择让JVM来解决什么时候才链接这个类。

如果你不想重新定义加载类的规则,也没有复杂的处理逻辑,只想在运行时能够加载自己指定的一个类,那么你可以用this.getClass().getClassLoader().loadClass("class")调用ClassLoader的loadClass方法以获取这个类的Class对象,这个loadClass还有重载方法,你同样可以决定在什么时候解析这个类。

ClassLoader是个抽象类,它还有很多子类,我们如果要实现自己的ClassLoader,一般都会继承URLClassLoader这个子类,因为这个类已经帮我们实现了大部分工作,我们只需要在适当的地方做些修改就行。

ClassLoader的等级加载机制(上级委托接待机制)

如何保证不同等级的会员通过不同的会员接待室进入会场?保证每个会员不会走错接待室,并且每个会员只能被一个接待室接待,从而保持接待的一致性。如何设计这个接待规则?

ClassLoader就设计了这样一种接待机制,就是上级委托接待机制。具体是这样的:任何一个会员到达任何一个接待室时,这个接待室首先会检查这个会员是否已经被自己接待过,如果接待过,那么拒绝本次接待,也就是不再发入会证明了,如果没有接待过,那么会向上询问这个会员是否应该在上一级的更高级别的接待室接待,上级接待室会根据它们的接待规则,检查这个会员是否已经被接待过,如果已经接待过,同样的处理方式,将已经接待过的结果反馈给下一级,如果也没有接待过,再向更高一级接待室转发接待请求,更高一级也还是同样的处理方法,直到有一级接待室接待或者告诉它下一级这个会员不是自己接待的这个结果;如果这个会员来到的这个接待室得到它上一级的接待室反馈认为这个会员没有被接待,并且也不应该有它们接待,这个接待室会正式接待这个会员,并发给他入会证明,这个会员就会被定义为这个接待室等级的会员。

整个JVM平台提供三层ClassLoader,这三层ClassLoader可以分为两种类型,可以理解为--->为接待室服务的接待室和为会员服务的接待室两种。

(1)Bootstrap ClassLoader,这个ClassLoader就是接待室服务自身的,它主要加载JVM自身工作需要的类。这个ClassLoader完全是由JVM自己控制的,需要加载哪个类,怎么加载都有JVM自己控制,别人也访问不到这个类,所以这个ClassLoader是不遵守前面介绍的加载规则的,它仅仅是一个类的加载工具而已,既没有更高一级的父加载器,也没有子加载器。

(2)ExtClassloader,这个类加载器有点特殊,它是JVM自身的一部分,但是它的血统也不是很纯正,它并不是JVM亲自实现的,我们可以理解为这个类加载器是那些与这个大会合作单位的员工会员,这些会员既不是JVM内部的,也和普通的外部会员不同,所以就由这个类加载器来加载,它服务的特定目标在System.getProperty("java.ext.dirs")目录下。

(3)AppClassLoader,这个类加载器就是专门为接待会员服务的,它的父类是ExtClassLoader。它服务的目标是广大普通会员,所有在System.,getProperty("java.class.path")目录下的类都可以被这个类加载器加载,这个目录就是我们常用到的classpath。

JVM加载class文件到内存有两种方式:

(1) 隐式加载:所谓隐式加载就是不通过在代码里调用ClassLoader来加载需要的类,而是通过JVM来自动加载需要的类到内存的方式。例如,当我们在类中继承或者引用某个类时,JVM在解析当前这个类时发现引用的类不在内存中,那么就会自动将这些类加载到内存中。

(2)显示加载:相反的显示加载就是我们在代码中通过ClassLoader类来加载一个类的方式。

如何加载class文件

ClassLoader加载一个class文件到JVM时需要经过的步骤。
第一个阶段是找到.class文件并把这个文件包含的字节码加载到内存中。
第二个阶段又可以分为三个步骤,分别是字节码验证、Class类数据结构分析及相应的内存分配和最后的符号表的链接。
第三阶段是类中静态属性和初始化赋值,以及静态块的执行。
加载字节码到内存 其实在ClassLoader抽象类中并没有定义如何去加载,如何去找到指定类并把它的字节码加载到内存。需要实现findClass()方法。子类URLClassLoader是如何实现findClass()的,在URLClassLoader中通过一个URLClassPath类取得要加载的class文件字节流,而这个URLClassPath定义了到哪里去找这个class文件,如果找到了class文件,再读取它的byte字节流,通过调用defineClass()方法来创建类对象。
</p>

常见加载类错误分析

在执行Java程序是经常会碰到ClassNotFoundException和NoClassDefFoundError两个异常,它们都和类加载有关。
(1) ClassNotFoundException

当JVM要加载指定文件的字节码到内存时,并没有找到这个文件对应的字节码。解决的办法就是检查在当前的classpath目录下有没有指定的文件的存在,如果不知道当前classpath路径,可以通过命令获取:

this.getClass().getClassLoader().getResource("").toString()  

(2) NoClassDefFoundError

常出现在第一次使用命令执行java类    例如:

java -cp example.jar Example

这里是因为在命令行中没有加类的包名   正确的写法:

 java -cp example.jar  net.aaa.Example

(3) UnstatusfiledLinkError
(4) ClassCastException
(5) ExceptionInInitializerError

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

推荐阅读更多精彩内容