其实,双亲委派模型并不复杂。自定义类加载器也不难!随便从网上搜一下就能搜出一大把结果,而后copy
一下就能用。可是,若是每次想自定义类加载器就必须搜一遍别人的文章,而后复制,这样显然不行。但是自定义类加载器又不常常用,时间久了容易忘记。相信你常常会记不太清loadClass
、findClass
、defineClass
这些函数我到底应该重写哪个?它们主要是作什么的?本文大体分析了各个函数的流程,目的就是让你看完以后,难以忘记!或者说,延长你对自定义类加载器的记忆时间!随时随地想自定义就自定义!java
关于双亲委派模型,网上的资料有不少。我这里只简单的描述一下,就当是复习。bootstrap
首先,先要知道什么是类加载器。简单说,类加载器就是根据指定全限定名称将class
文件加载到JVM
内存,转为Class
对象。若是站在JVM
的角度来看,只存在两种类加载器:数组
启动类加载器(
Bootstrap ClassLoader
):由C++
语言实现(针对HotSpot
),负责将存放在<JAVA_HOME>\lib
目录或-Xbootclasspath
参数指定的路径中的类库加载到内存中。函数其余类加载器:由
Java
语言实现,继承自抽象类ClassLoader
。如:this
- 扩展类加载器(
Extension ClassLoader
):负责加载<JAVA_HOME>\lib\ext
目录或java.ext.dirs
系统变量指定的路径中的全部类库。- 应用程序类加载器(
Application ClassLoader
)。负责加载用户类路径(classpath
)上的指定类库,咱们能够直接使用这个类加载器。通常状况,若是咱们没有自定义类加载器默认就是用这个加载器。
双亲委派模型工做过程是:若是一个类加载器收到类加载的请求,它首先不会本身去尝试加载这个类,而是把这个请求委派给父类加载器完成。每一个类加载器都是如此,只有当父加载器在本身的搜索范围内找不到指定的类时(即ClassNotFoundException
),子加载器才会尝试本身去加载。加密
为何须要双亲委派模型呢?假设没有双亲委派模型,试想一个场景:spa
黑客自定义一个
java.lang.String
类,该String
类具备系统的String
类同样的功能,只是在某个函数稍做修改。好比equals
函数,这个函数常用,若是在这这个函数中,黑客加入一些“病毒代码”。而且经过自定义类加载器加入到JVM
中。此时,若是没有双亲委派模型,那么JVM
就可能误觉得黑客自定义的java.lang.String
类是系统的String
类,致使“病毒代码”被执行。code
而有了双亲委派模型,黑客自定义的java.lang.String
类永远都不会被加载进内存。由于首先是最顶端的类加载器加载系统的java.lang.String
类,最终自定义的类加载器没法加载java.lang.String
类。orm
或许你会想,我在自定义的类加载器里面强制加载自定义的java.lang.String
类,不去经过调用父加载器不就行了吗?确实,这样是可行。可是,在JVM
中,判断一个对象是不是某个类型时,若是该对象的实际类型与待比较的类型的类加载器不一样,那么会返回false。对象
举个简单例子:
ClassLoader1
、ClassLoader2
都加载java.lang.String
类,对应Class一、Class2对象。那么Class1
对象不属于ClassLoad2
对象加载的java.lang.String
类型。
双亲委派模型的原理很简单,实现也简单。每次经过先委托父类加载器加载,当父类加载器没法加载时,再本身加载。其实ClassLoader
类默认的loadClass
方法已经帮咱们写好了,咱们无需去写。
loadClass
默认实现以下:
public Class<?> loadClass(String name) throws ClassNotFoundException { return loadClass(name, false); }
再看看loadClass(String name, boolean resolve)
函数:
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; } }
从上面代码能够明显看出,loadClass(String, boolean)
函数即实现了双亲委派模型!整个大体过程以下:
- 首先,检查一下指定名称的类是否已经加载过,若是加载过了,就不须要再加载,直接返回。
- 若是此类没有加载过,那么,再判断一下是否有父加载器;若是有父加载器,则由父加载器加载(即调用
parent.loadClass(name, false);
).或者是调用bootstrap
类加载器来加载。- 若是父加载器及
bootstrap
类加载器都没有找到指定的类,那么调用当前类加载器的findClass
方法来完成类加载。
话句话说,若是自定义类加载器,就必须重写findClass
方法!
findClass
的默认实现以下:
protected Class<?> findClass(String name) throws ClassNotFoundException { throw new ClassNotFoundException(name); }
能够看出,抽象类ClassLoader
的findClass
函数默认是抛出异常的。而前面咱们知道,loadClass
在父加载器没法加载类的时候,就会调用咱们自定义的类加载器中的findeClass
函数,所以咱们必需要在loadClass
这个函数里面实现将一个指定类名称转换为Class
对象.
若是是是读取一个指定的名称的类为字节数组的话,这很好办。可是如何将字节数组转为Class
对象呢?很简单,Java
提供了defineClass
方法,经过这个方法,就能够把一个字节数组转为Class对象啦~
defineClass
主要的功能是:
将一个字节数组转为
Class
对象,这个字节数组是class
文件读取后最终的字节数组。如,假设class
文件是加密过的,则须要解密后做为形参传入defineClass
函数。
defineClass
默认实现以下:
protected final Class<?> defineClass(String name, byte[] b, int off, int len) throws ClassFormatError { return defineClass(name, b, off, len, null); }
上一节所提的函数调用过程以下:
首先,咱们定义一个待加载的普通Java
类:Test.java
。放在com.huachao.cl
包下:
package com.huachao.cl; public class Test { public void hello() { System.out.println("恩,是的,我是由 " + getClass().getClassLoader().getClass() + " 加载进来的"); } }
注意:
若是你是直接在当前项目里面建立,待
Test.java
编译后,请把Test.class
文件拷贝走,再将Test.java
删除。由于若是Test.class
存放在当前项目中,根据双亲委派模型可知,会经过sun.misc.Launcher$AppClassLoader
类加载器加载。为了让咱们自定义的类加载器加载,咱们把Test.class文件放入到其余目录。
在本例中,咱们Test.class文件存放的目录以下:
接下来就是自定义咱们的类加载器:
import java.io.FileInputStream; import java.lang.reflect.Method; public class Main { static class MyClassLoader extends ClassLoader { private String classPath; public MyClassLoader(String classPath) { this.classPath = classPath; } private byte[] loadByte(String name) throws Exception { name = name.replaceAll("\\.", "/"); FileInputStream fis = new FileInputStream(classPath + "/" + name + ".class"); int len = fis.available(); byte[] data = new byte[len]; fis.read(data); fis.close(); return data; } protected Class<?> findClass(String name) throws ClassNotFoundException { try { byte[] data = loadByte(name); return defineClass(name, data, 0, data.length); } catch (Exception e) { e.printStackTrace(); throw new ClassNotFoundException(); } } }; public static void main(String args[]) throws Exception { MyClassLoader classLoader = new MyClassLoader("D:/test"); Class clazz = classLoader.loadClass("com.huachao.cl.Test"); Object obj = clazz.newInstance(); Method helloMethod = clazz.getDeclaredMethod("hello", null); helloMethod.invoke(obj, null); } }
最后运行结果以下:
恩,是的,我是由 class Main$MyClassLoader 加载进来的