JAVA的ClassLoader

作者: WANGGGGG | 来源:发表于2019-05-06 23:47 被阅读10次

    前言

    学完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);
    }
    

    相关文章

      网友评论

        本文标题:JAVA的ClassLoader

        本文链接:https://www.haomeiwen.com/subject/jfjngqtx.html