JAVA的ClassLoader

前言

学完Class类,知道了每一个.java文件在编译后会保存成.class文件,文件中保存了该类的具体信息,然后我就好奇JVM怎么加载的类,所以就必须要探索一下ClassLoader了

一、背景知识

1.1 类加载器种类

类加载器主要有三种:
(1)Bootstrap ClassLoader 根加载器,用于加载java.包下面的类
(2)Extension ClassLoader 补充类加载器,用于加载javax.
路径的类,这个包下面的类主要是对java包下面类的功能补充
(3)Application ClassLoader 应用类的加载,加载用户路径下的类
如果上面的类加载器都不满足你的需求,可以继承ClassLoader,实现自己的类加载器,后面会介绍自定义类加载器需要实现哪几个方法。

1.2 双亲委派模式

双亲委派的过程:需要加载一个类,首先需要判断这个类是否已经被加载,如果没被加载首先使用父类加载器,然后再使用自己的类加载器来加载类。这样就可以防止被入侵的风险,假设有人自定义了一个你已经有的类,然后覆盖了你的其中一个方法,然后在里面做了一些危险的操作,如果这个类被加载进来,并覆盖了原来你真实的类,这结果就很尴尬了。
源码实现过程如下:

/**
     * 加载类
     * @param name 类的全路径名
     * @param resolve  检测最终有没有加载到类
     * @return
     * @throws ClassNotFoundException
     */
    protected Class<?> loadClass(String name, boolean resolve)
            throws ClassNotFoundException
    {
        // 同步锁,加载类需要先获取该类对应的锁
        synchronized (getClassLoadingLock(name)) {
            // 首先检查是否这个类已经被加载过了
            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
                }
                // 如果父加载器加载失败,则使用当前加载器来加载类信息
                // findClass本身是一个空的方法,需要加载器类去实现
                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
                    // 记录父加载器加载类所使用的时间
                    PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    // 加载计数+1
                    PerfCounter.getFindClasses().increment();
                }
            }
            // 加载最终有没有加载到类,如果c还是空的就抛出异常
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

二、如何自定义ClassLoader

2.1 findClass
这个方法的意图是输入类全路径名,然后加载得到类的Class对象,但是这里是空的,所以需要继承一下ClassLoader然后覆写一下。

protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
    }

2.2 defineClass方法
可以通过defineClass方法获取Class对象

/**
     * 输入类的字节信息,输出class对象
     *
     * @param name             类的全路径名
     * @param b                类的字节数组
     * @param protectionDomain 类的保护域,里面可以设置安全信息,一般可以传null
     * @return
     * @throws ClassFormatError
     */
    protected final Class<?> defineClass(String name, java.nio.ByteBuffer b,
                                         ProtectionDomain protectionDomain)
            throws ClassFormatError {
        // 类字节信息长度
        int len = b.remaining();

        // 判断是否为直接buffer
        if (!b.isDirect()) {
            if (b.hasArray()) {
                // 直接通过类字节数组定义类,这里不是递归,而是调用本地的另外一个同名方法,源码在下面展出
                return defineClass(name, b.array(),
                        b.position() + b.arrayOffset(), len,
                        protectionDomain);
            } else {
                // 将byteBuffer中的字节数组加载到byte中
                byte[] tb = new byte[len];
                b.get(tb);
                // 这里不是递归,而是调用本地的另外一个同名方法,源码在下面展出
                return defineClass(name, tb, 0, len, protectionDomain);
            }
        }
        // 做一些前置处理
        // (1)检查名称中是否含有“/”,或是以“[”开头,这两个开头直接抛出异常
        // (2)检查是否以java.开头,这个是禁止的,也会抛出异常,因为会有安全问题
        // (3)检查锁信息
        protectionDomain = preDefineClass(name, protectionDomain);
        String source = defineClassSourceLocation(protectionDomain);
        // 调用本地方法,使用直接内存的字节数组获得类信息
        Class<?> c = defineClass2(this, name, b, b.position(), len, protectionDomain, source);
        // 做一些后置处理,如设置一下,类对应的包名
        postDefineClass(c, protectionDomain);
        return c;
    }

另一种方式更为直接点,利用类信息的字节数组(非byteBuffer,和上面的方法是有区别的),得到类的Class对象

 protected final Class<?> defineClass(String name, byte[] b, int off, int len,
                                         ProtectionDomain protectionDomain)
        throws ClassFormatError
    {
        protectionDomain = preDefineClass(name, protectionDomain);
        String source = defineClassSourceLocation(protectionDomain);
        Class<?> c = defineClass1(this, name, b, off, len, protectionDomain, source);
        postDefineClass(c, protectionDomain);
        return c;
    }

2.3 简单的实现类的加载
覆写一下findClass调用defineClass

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

推荐阅读更多精彩内容