自定义类加载器

import lombok.extern.slf4j.Slf4j;
import org.apache.commons.io.IOUtils;

import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Objects;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * @title CustomClassLoader
 * @description 自定义类加载器
 * author zzw
 * version 1.0.0
 * create 2024/8/18 01:34
 **/
@Slf4j
public class CustomClassLoader extends URLClassLoader {

    private JarFile jarFile;
    private ClassLoader parent;

    public CustomClassLoader(URL[] urls, JarFile jarFile, ClassLoader parent) {
        super(urls, parent);
        this.jarFile = jarFile;
        this.parent = parent;
    }

    public CustomClassLoader(URL[] urls, ClassLoader parent) {
        super(urls, parent);
    }

    /**
     * 将类的全限定名替换成类路径
     * com.doudou.thread.Test003 -> com/doudou/thread/Test003.class
     *
     * @param name 类的全限定名
     * @return
     */
    private static String classNameToJarEntry(String name) {
        String classPath = name.replaceAll("\\.", "\\/");
        return new StringBuilder(classPath).append(".class").toString();
    }

    /**
     * 重新loadClass方法,按照类包路径加载Class到JVM
     *
     * @param name    类全限定名
     * @param resolve
     * @return
     * @throws ClassNotFoundException
     */
    @Override
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        // 自定义类的加载规则,和findClass方法一起组合打破双亲委派
        // 如果类所在的包是以com.xx或com.yyy起始的,使用自定义加载方法进行类加载
        if (name.startsWith("com.xx") || name.startsWith("com.yyy")) {
            return this.findClass(name);
        }
        // 其它的使用双亲委派机制进行类加载
        return super.loadClass(name, resolve);
    }


    /**
     * 进行类的查找以及加载
     *
     * @param name 类的全限定名
     * @return 加载到的类
     * @throws ClassNotFoundException
     */
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        Class clazz = null;
        try {
            // 将类的全限定名转换成该类在类路径(jar包)中的路径
            String jarEntryName = classNameToJarEntry(name);
            // 自定义加载类是加载指定jar中的类,因此JarFile不存在时直接返回null
            if (Objects.isNull(jarFile)) {
                return clazz;
            }
            // 在jar包中找到指定类的信息
            JarEntry jarEntry = jarFile.getJarEntry(jarEntryName);
            // 找到类信息的情况下,将类信息转换为字节数组,并加载成class
            if (Objects.nonNull(jarEntry)) {
                InputStream inputStream = jarFile.getInputStream(jarEntry);
                byte[] bytes = IOUtils.toByteArray(inputStream);
                clazz = defineClass(name, bytes, 0, bytes.length);
            }
        } catch (IOException e) {
            log.info("Custom classloader load calss {} failed", name);
        }
        return clazz;
    }
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值