枯燥的JVM - 类加载机制

类加载器负责将 Java 类文件加载到 Java 虚拟机。
只有当类被加载进虚拟机内存,才能使用对应的类。

在 Java 中,类加载过程大概分为以下几步:

  1. 通过全限类名获取类文件字节数组。可来自本地文件、jar 包、网络等。
  2. 在方法区/元空间保存类的描述信息、静态属性。
  3. 在 JVM 堆中生成一个对应的 java.lang.Class 对象。

具体的加载过程为:


image.png

加载:在硬盘上查找并通过IO读入字节码文件,使用到类时才会加载,例如调用类的 main() 方法,new 对象等等,在加载阶段会在内存中生成一个代表这个类的 java.lang.Class 对象,作为方法区这个类的各种数据的访问入口。
验证:校验字节码文件的正确性。
准备:给类的静态变量分配内存,并赋予默认值。
解析:将符号引用替换为直接引用,该阶段会把一些静态方法(符号引用,比如 main() 方法)替换为指向数据所存内存的指针或句柄等(直接引用),这是所谓的静态链接过 程(类加载期间完成),动态链接是在程序运行期间完成的将符号引用替换为直接引用。
初始化:对类的静态变量初始化为指定的值,执行静态代码块。

Java 默认提供三个类加载器,分别为:

  • Bootstrap ClassLoader
  • Extension ClassLoader
  • App ClassLoader

Bootstrap ClassLoader 负责加载Java基础类,主要是 %JRE_HOME%/lib/ 目录下的rt.jar、resources.jar、charsets.jar等。
Extension ClassLoader 负责加载Java扩展类,主要是 %JRE_HOME%/lib/ext 目录下的jar。
App ClassLoader 负责加载当前应用的ClassPath中的所有类。

类加载器加载的流程如下:


image.png

先简单的打印一下默认的加载器:

public class Hello {
    public static void main(String[] args) {
        ClassLoader classLoader = Hello.class.getClassLoader();
        System.out.println(classLoader.getParent().getParent());
        System.out.println(classLoader.getParent());
        System.out.println(classLoader);
    }
}

打印结果为:

null
sun.misc.Launcher$ExtClassLoader@61bbe9ba
sun.misc.Launcher$AppClassLoader@18b4aac2

由于引导类加载器由 c++ 实现 故为 null(加载器之间不是真正的继承关系)

类加载器初始化过程:

创建 JVM 启动器实例 sun.misc.Launcher。

sun.misc.Launcher 初始化使用了单例模式设计,保证一个 JVM 虚拟机内只有一个 sun.misc.Launcher 实例。

public Launcher() {
        Launcher.ExtClassLoader var1;
        try {
            var1 = Launcher.ExtClassLoader.getExtClassLoader();  // 创建扩展类加载器
        } catch (IOException var10) {
            throw new InternalError("Could not create extension class loader", var10);
        }

        try {
            this.loader = Launcher.AppClassLoader.getAppClassLoader(var1);  // 创建应用类加载器,并把自己的parent引用指向扩展类加载器
        } catch (IOException var9) {
            throw new InternalError("Could not create application class loader", var9);
        }
        ...其它代码省略
}

在Launcher构造方法内部,其创建了两个类加载器,分别是Launcher.ExtClassLoader (扩展类加载器) 和 Launcher.AppClassLoader (应用类加载器)。
JVM 默认使用 Launcher 的 getClassLoader() 方法返回的类加载器 AppClassLoader 的实例加载我们的应用程序。

双亲委派机制

加载器在加载某个类时会先委托父加载器寻找目标类,找不到再委托上层父加载器加载,如果所有父加载器在自己的加载类路径下都找不到目标类,则在自己的类加载路径中查找并载入目标类。部分实现如下:

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) {
                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
                    c = findClass(name);
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

为什么要设计双亲委派机制?

沙箱安全机制:自己写的 java.lang.String.class 类不会被加载,这样便可以防止核心 API库被随意篡改。
避免类的重复加载:当父亲已经加载了该类时,就没有必要子 ClassLoader 再加载一 次,保证被加载类的唯一性。
若想打破双亲加载机制,重写此加载方法,实现自己的加载逻辑,不委派给双亲加载即可。

全盘负责委托机制

“全盘负责”是指当一个 ClassLoder 装载一个类时,除非显示的使用另外一个 ClassLoder ,该类所依赖及引用的类也由这个 ClassLoder 载入。

自定义类加载器

自定义类加载器则可以实现额外的需求,例如:

从网络文件加载类。
从任意目录加载类。
对字节码文件做加密处理,由自定义类加载器做解密。

下面是继承了 URLClassLoader 的一个简单自定义类加载器

public class MyClassLoader extends URLClassLoader {
    private String classPath;
    public MyClassLoader(URL[] urls){
        super(urls); 
    }
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, MalformedURLException {
        File file = new File("/opt"); // 自定义的加载根目录
        URL[] URLs = new URL[] {file.toURI().toURL()};
        MyClassLoader myClassLoader = new MyClassLoader(URLs);
        Class<?> aClass = myClassLoader.loadClass("com.app.Hello"); // java文件的具体地址
        Object object = aClass.newInstance(); // 通过反射来调用对象里的方法
        Method method = aClass.getDeclaredMethod("sayHello", String.class);
        method.invoke(object, "lilei");
        System.out.println(object.getClass().getClassLoader().getParent());
        System.out.println(object.getClass().getClassLoader());
    }
}

打印结果为:

hello lilei
sun.misc.Launcher$AppClassLoader@18b4aac2
com.app.MyClassLoader@61bbe9ba

URLClassLoader 类中主要帮我们实现了 ClassLoader 的 findClass 方法,实现代码如下:

protected Class<?> findClass(final String name)
        throws ClassNotFoundException
    {
        final Class<?> result;
        try {
            result = AccessController.doPrivileged(
                new PrivilegedExceptionAction<Class<?>>() {
                    public Class<?> run() throws ClassNotFoundException {
                        String path = name.replace('.', '/').concat(".class");
                        Resource res = ucp.getResource(path, false);
                        if (res != null) {
                            try {
                                return defineClass(name, res);
                            } catch (IOException e) {
                                throw new ClassNotFoundException(name, e);
                            }
                        } else {
                            return null;
                        }
                    }
                }, acc);
        } catch (java.security.PrivilegedActionException pae) {
            throw (ClassNotFoundException) pae.getException();
        }
        if (result == null) {
            throw new ClassNotFoundException(name);
        }
        return result;
    }

实现自定义类加载器的主要步骤为:

继承 ClassLoader 类。如果只是从目录或者jar包加载类,也可以选择继承 URLClassLoader 类。
重写 findClass 方法。
在重写的 findClass 方法中,无论用何种方法,获取类文件对应的字节数组,然后调用 defineClass 方法转换成类实例。

需要注意下,即使自己自定义了类加载器,并重写了 loadClass 方法打破双亲委派机制,也无法篡改 java 核心类,最后来看一下 ClassLoader 中 defineClass 的部分校验代码:

private ProtectionDomain preDefineClass(String name, ProtectionDomain pd) {
        if (!checkName(name))
            throw new NoClassDefFoundError("IllegalName: " + name);
        // 防止篡改其核心类
        if ((name != null) && name.startsWith("java.")) {
            throw new SecurityException
                ("Prohibited package name: " +
                 name.substring(0, name.lastIndexOf('.')));
        }
        if (pd == null) {
            pd = defaultDomain;
        }
        if (name != null) checkCerts(name, pd.getCodeSource());
        return pd;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342