类加载器-自定义类加载器(打破双亲委托机制加载)

本文介绍了如何通过创建自定义类加载器`SimpleClassLoader`来打破Java的双亲委托机制。在`loadClass`方法中,首先尝试使用自定义类加载器加载类,如果失败,则回退到系统类加载器加载以`java.`开头的类。示例代码展示了如何加载指定目录下的类文件,并提供了测试类`SimpleClassLoaderTest`来验证加载过程。
摘要由CSDN通过智能技术生成

打破双亲委托机制加载

对一些系统的包如:java.lang包就只能用系统类加载器

对象

package com.ln.concurrent.classloader.chapter5;

/**
 * @ProjectName: java-concurrency
 * @Package: com.ln.concurrent.classloader.chapter5
 * @Name:SimpleObject
 * @Author:linianest
 * @CreateTime:2021/1/6 16:18
 * @version:1.0
 * @Description TODO: 对象
 */
public class SimpleObject {
}

自定义类加载器:重写父类的loadClass方法,可以打破双亲委托机制

package com.ln.concurrent.classloader.chapter5;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

/**
 * @ProjectName: java-concurrency
 * @Package: com.ln.concurrent.classloader
 * @Name:SimpleClassLoader
 * @Author:linianest
 * @CreateTime:2021/1/6 16:06
 * @version:1.0
 * @Description TODO: 自定义类加载器
 */
public class SimpleClassLoader extends ClassLoader {
    // 默认的加载路径
    private static final String DEFULT_DIR = "E:\\workspace\\java-concurrency\\classLoader\\target\\classes";

    private String dir = DEFULT_DIR;

    // 类加载器的名称
    private String classLoaderName;

    public SimpleClassLoader() {
        super();
    }

    public SimpleClassLoader(String classLoaderName) {
        super();
        this.classLoaderName = classLoaderName;
    }

    // 指定该类加载器的父亲是哪个
    public SimpleClassLoader(String classLoaderName, ClassLoader parent) {
        super(parent);
        this.classLoaderName = classLoaderName;
    }


    public String getDir() {
        return dir;
    }

    public void setDir(String dir) {
        this.dir = dir;
    }



    /**
     *  如果包含java的内容,使用系统类加载器
     *  否则先用自定义的类加载器,加载不出来再用系统类加载器
     * @param name
     * @param resolve
     * @return
     * @throws ClassNotFoundException
     */
    @Override
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        Class<?> clazz = null;
        if (name.startsWith("java.")) {
            try {
                final ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
                clazz = systemClassLoader.loadClass(name);
                if (clazz != null) {
                    if (resolve) {
                        resolveClass(clazz);
                        return clazz;
                    }
                }
            } catch (Exception e) {
                //ignore
            }
        }
        try {
            clazz = this.findClass(name);
        } catch (ClassNotFoundException e) {
        }
        if (null == clazz && getParent() != null) {
            getParent().loadClass(name);
        }
        return clazz;
    }

    /**
     * 返回找的class文件字节数组
     *
     * @param name
     * @return
     * @throws ClassNotFoundException
     */
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        final String classpath = name.replace(".", "/");
        final File classfile = new File(dir, classpath + ".class");
        if (!classfile.exists()) {
            throw new ClassNotFoundException("The class " + name + " not found under directory [" + dir + "]");
        }
        // 将文件读取到字节数组中
        byte[] classBytes = loadClassBytes(classfile);
        if (null == classBytes || classBytes.length == 0) {
            throw new ClassNotFoundException("load the class " + name + " failed");
        }

        return this.defineClass(name, classBytes, 0, classBytes.length);
    }

    /**
     * 将文件读取到内存字节数组并加密后,输出流中
     *
     * @param classfile
     * @return
     */
    private byte[] loadClassBytes(File classfile) {
        try (final ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
             final FileInputStream fileInputStream = new FileInputStream(classfile);) {
            final byte[] buffer = new byte[1024];
            int len;
            // 将读取的数据流到缓冲区中并输出
            while ((len = fileInputStream.read(buffer)) != -1) {
                byteArrayOutputStream.write(buffer, 0, len);
            }
            byteArrayOutputStream.flush();
            return byteArrayOutputStream.toByteArray();
        } catch (IOException e) {
            e.printStackTrace();
            return null;
        }
    }
}

打破双亲先加载的机制

package com.ln.concurrent.classloader.chapter5;

import com.ln.concurrent.classloader.chapter3.MyClassLoader;

import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * @ProjectName: java-concurrency
 * @Package: com.ln.concurrent.classloader.chapter5
 * @Name:SimpleClassLoaderTest
 * @Author:linianest
 * @CreateTime:2021/1/6 16:09
 * @version:1.0
 * @Description TODO: 打破双亲先加载的机制
 */
public class SimpleClassLoaderTest {

    public static void main(String[] args) throws ClassNotFoundException{
        final SimpleClassLoader simpleClassLoader = new SimpleClassLoader("E:\\workspace\\java-concurrency\\classLoader\\target\\classes");

        final Class<?> loadClass = simpleClassLoader.loadClass("com.ln.concurrent.classloader.chapter5.SimpleObject");
        System.out.println(simpleClassLoader);
        System.out.println(loadClass.getClassLoader());


    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值