Java类加载机制

参考https://mp.weixin.qq.com/s/x3Z6srrlTJsPzPS8QMSyWg

首先能看到 ExtClassLoader 确实是 AppClassLoader 的双亲,
不过却没有看到 BootstrapClassLoader。
事实上,上文就提过, BootstrapClassLoader比较特殊,它是由 JVM 内部实现的,所以 ExtClassLoader.getParent() = null。
package com.company.类加载.classloader;

import javafx.scene.Parent;

/**
 * Created by Administrator on 2018/7/11 0011.
 */
public class Test {
    public static void main(String[] args) {
        //loadClass();
        printParent();

    }

    private static void loadClass() {
        try {
            Class<?> clazz = Class.forName("com.company.类加载.classloader.MusicPlayer");
            ClassLoader classLoader = clazz.getClassLoader();
            System.out.printf("ClassLoader is %s", classLoader.getClass().getSimpleName());
        } catch (Exception e) {
            System.out.println("error");
            System.out.println(e.getMessage());
        }

    }

    private static void printParent() {
        try {
            Class<?> clazz = Class.forName("com.company.类加载.classloader.MusicPlayer");
            ClassLoader classLoader = clazz.getClassLoader();
            System.out.printf("currentClassLoader is %s\n", classLoader.getClass().getSimpleName());
            //ExtClassLoader.getParent() = null 
            while (classLoader.getParent() != null) {
                classLoader = classLoader.getParent();
                System.out.printf("Parent is %s\n", classLoader.getClass().getSimpleName());
            }
        } catch (Exception e) {
            System.out.println("error");
            System.out.println(e);
        }

    }

}

源码阅读

//类加载源码阅读
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
            // 1. 检查是否曾加载过
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        // 优先让 parent 加载器去加载
                        c = parent.loadClass(name, false);
                    } else {
                        // 如无 parent,表示当前是 BootstrapClassLoader,调用 native 方法去 JVM 加载
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
                if (c == null) {                // 如果 parent 均没有加载到目标class,调用自身的 findClass() 方法去搜索
                    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;
        }
    }
    // BootstrapClassLoader 会调用 native 方法去 JVM 加载private native Class<?> findBootstrapClass(String name);

从网络加载类

public class NetworkClassLoader extends ClassLoader {
        @Override
        protected Class<?> findClass(String name) throws ClassNotFoundException {
            byte[] classData = downloadClassData(name); // 从远程下载
            if (classData == null) {
                super.findClass(name); // 未找到,抛异常
            } else {
                return defineClass(name, classData, 0, classData.length); // convert class byte data to Class<?> object
            }
            return null;
        }
private byte[] downloadClassData(String name) {        // 从 localhost 下载 .class 文件
            String path = "http://localhost" + File.separatorChar + "java" + File.separatorChar + name.replace('.', File.separatorChar) + ".class";
            try {
                URL url = new URL(path);
                InputStream ins = url.openStream();
                ByteArrayOutputStream baos = new ByteArrayOutputStream();
                int bufferSize = 4096;
                byte[] buffer = new byte[bufferSize];
                int bytesNumRead = 0;
                while ((bytesNumRead = ins.read(buffer)) != -1) {
                    baos.write(buffer, 0, bytesNumRead); // 把下载的二进制数据存入 ByteArrayOutputStream
                }
                return baos.toByteArray();
            } catch (Exception e) {
                e.printStackTrace();
            }
            return null;
        }

        public String getName() {
            System.out.printf("Real NetworkClassLoader\n");
            return "networkClassLoader";
        }
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值