如何把自定义类加载器使用?

2022-09-01 07:51:03

大家好,感谢您的关注!我有一个问题,必须既简单又明显,但我被困住了。

我想通过自定义 ClassLoader 提供动态创建的 Java 类,供第三方库使用。

现在我的问题是:当我不直接自己加载它们时,如何将我的自定义类加载器设置为用于加载这些类?

我以为当我使用我的ClassLoader加载某个类时,它变成了这个类的ClassLoader,从该类加载的所有类都将通过我的ClassLoader进行引导。

我按照这个官方教程创建了一个自定义的ClassLoader:http://java.sun.com/developer/onlineTraining/Security/Fundamentals/magercises/ClassLoader/help.html

public class DynamicClassloader extends ClassLoader {

    private Map<String, Class<?>> classesMap = new HashMap<String, Class<?>>();

    public DynamicClassloader(ClassLoader parent) {
        // Also tried super(parent);
        super(sun.misc.Launcher.getLauncher().getClassLoader());
    }

    // Adding dynamically created classes
    public void defineClass(String name, Class<?> clazz) {
        classesMap.put(name, clazz);
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        // load from parent
        Class<?> result = findLoadedClass(name);
        if (result != null) {
            return result;
        }
        try {
            result = findSystemClass(name);
        } catch (Exception e) {
            // Ignore these
        }
        if (result != null) {
            return result;
        }
        result = classesMap.get(name);
        if (result == null) {
            throw new ClassNotFoundException(name);
        }
        return result;
    }
}

我想在代码中的其他地方使用它,如下所示:

ClassLoader thisClassLoader = this.getClass().getClassLoader();
((DynamicClassloader) thisClassLoader).defineClass(className, dynClass);

现在我的问题是,当我调用第三方库类时,父类加载器找到这个类(因为它在类路径上)并成为它的类加载器。由于父类加载器不知道我的自定义类加载器,因此它实际上已被停用,无法强制转换为DynamicClassLoader。findSystemClass(name)this.getClass().getClassLoader()

另一种方法是通过JVM参数将我的ClassLoader设置为系统ClassLoader。但这给了我一个StackOverflowError:-Djava.system.class.loader=my.DynamicClassloader

    ...
at de.unisaarland.cs.st.DynamicClassloader.findClass(DynamicClassloader.java:39)
at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
at java.lang.ClassLoader.findSystemClass(ClassLoader.java:916)
at de.unisaarland.cs.st.DynamicClassloader.findClass(DynamicClassloader.java:39)
    ...

这一定很容易做到,但我现在没有想法了......任何帮助都非常感谢!


答案 1

不确定我是否理解这个问题,你有一个第三方库,你希望它使用你的类加载器来加载类。

如果你很幸运,第三方库使用线程上下文类加载器,您可以使用 设置,在同一线程中,您可以使用...Thread.currentThread().setContextClassLoader(myClassLoader)Thread.currentThread().getContextClassLoader()

另一点,但不确定它在你的上下文中是否重要,是你也可以编写一个父最后一个类加载器,它将在委托给其父类之前尝试加载类(而不是先尝试委托)

在您的评论后编辑:

如果您的库不依赖于线程上下文类装入器,那么parent_last类装入器将会有所不同,那么您必须使用父级-最后一个类装入器装入库,从而将类装入器设置为库的类装入器,而不是其父装入器(类装入器的父级)...

您也可以使用父级优先行为制作类加载器,但对于您的第三方库...

还有一个关于类装入器的好链接...


答案 2

推荐