classloader

前言

最近看了很多关于classloader的文章,想了解一下classload具体的工作原理。因为前几日跟盆友聊天,说让我研究研究Android的插件技术。随后我搜索一些相关的资料,其中有一篇博文讲到其底层的技术就是ClassLoader,所以要对java的ClassLoader有一定的了解。当然,随着最近几天的阅读,也对插件技术有了一定程度的认知。各种框架,各种技术流派在国内可谓层出不穷,阿里的ATLAS,携程的DynamicAPK,奇虎360的DriodPlugin等等。但似乎很少看到国外对此技术的热衷,虽然Google给了一个MultiDex,但谷歌还是不建议这么做,尤其随着ReactNative的兴起,通过JsPath即可实现热修复(iOS),相必其可能是未来的发展方向。当然,通过对Android插件技术的学习,也是一个对Android四大组件(Activity,Service,Broadcast,Content Provider)充分认识的过程,毕竟背后隐藏的逻辑也逃不过这些基础的东西。基础中的基础那么可能就是ClassLoader了吧。

关于ClassLoader

ClassLoader从名字就能看出,类加载器。为什么要类加载器?这使我想起了.NET的程序集的概念,很像。一个Assembly.Load就能将一个DLL加载进来。很遗憾,没有对.NET程序集更深层次做挖掘,所以希望能对java的类加载机制有一定认识。我们知道引入一个类,只需要import java.io.File,没错import,为什么使用这个关键字就能加载?对于java的基础类库来说,虚拟机帮你做好了。这个就跟.NET一样,一个using就可以加载全局程序集,而要加载自定义的程序集,就必须在相同目录,即私有程序集。JAVA也一样,有安装java的时候自带的,也有你自己定义的,你可以把你的jar放在lib目录,也可以放在当前目录,随你。

编译的过程

JAVA属于解释型的编程语言,这个不用多解释吧,就是不编译成最终的目标平台的二进制,而是编译成中间语言,.NET叫IL,JAVA叫.class,叫啥无所谓,都一样。传统的编译过程是编译成目标文件,然后在对目标文件进行链接,但是对于JAVA而言,首先编译成的是字节码.class文件,在JVM加载class的时候,才进行链接。整个java的执行过程,按照Java Language Specification第12章的介绍,分为如下几个过程:

  1. JVM运行。
  2. 加载Main函数(启动类)
  3. 对目标类进行链接(验证,准备,保留)
  4. 初始化。
  5. 创建。
  6. 终结一个类
  7. 卸载。
  8. 程序退出。

三个类

Paste_Image.png
  • BootstrapClassLoader: 这个是native code写的,嵌入在jvm里,虚拟机启动的时候自动启动bootstrapclassloader,加载lib下的类库。
  • ExtensionClassLoader: 负责加载lib/ext里面的类库。
  • ApplicationClassLoader: 负责加载CLASSPATH里面的类库。
  • 三者的关系是:看上图,下面的继承上面的类。上面的为下面的parent。
  • 加载的顺序:先是BootstrapClassLoader加载,如果它没有找到,则ExtensionClassLoader尝试加载,如果它也没找到,则ApplicationClassLoader进行加载,都没找到,ClassNotFoundException。
  • 加载原理:双亲委托法,即child依次委托parent进行查找。(各博文均介绍如此)
  • ClassLoader使用loadClass方法加载一个类:
    protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }

                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    c = findClass(name);

                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

查看继承关系

public class LucasMainEntry {
    public static void main(String[] args) {
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        while(loader != null) {
            System.out.println(loader);
           loader = loader.getParent();
        }
    }
}
sun.misc.Launcher$AppClassLoader@2a139a55
sun.misc.Launcher$ExtClassLoader@7852e922
null

我们可以看到对应的父子关系。

加载过程中的问题

一个类如果被两个不同的loader加载,那么即便他们有相同的命名空间以及名称,JVM仍然认为他们不是同一个类。

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

推荐阅读更多精彩内容

  • 安卓插件化越来越流行,其中用到的技术不外乎加载外部的资源和加载外部的代码,关于加载外部资源我之前写过一篇文章《安卓...
    嘉伟咯阅读 1,608评论 5 14
  • 作者简介 原创微信公众号郭霖 WeChat ID: guolin_blog 本篇是fank909的第四篇投稿,详细...
    木木00阅读 1,601评论 1 14
  • 好久不写日志了。还记得以前有个姑娘说,看我的日志感觉很好。只是,你们都不会再看到。即使某些人还在。 有说过,人一直...
    赵业阅读 195评论 0 1
  • 画着好玩的画,继续小姐姐
    夏秋爱画阅读 236评论 0 0
  • 听梵音和我们平时去看演唱会一点都不一样。一个热闹狂热,一个安和宁静。 现场听梵音和自己在手机听也不一样,现场版更空...
    真墨流歌阅读 158评论 0 0