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