首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Java ClassLoader变更

Java ClassLoader变更
EN

Stack Overflow用户
提问于 2012-04-17 21:42:39
回答 3查看 6.7K关注 0票数 7

我有一些A

代码语言:javascript
复制
public class A {
    public A(String str) {
        System.out.println("Create A instance: " + str);
    }

    public void methodA() {
        System.out.println("#methodA1()");
    }
}

和我的类加载器实现:

代码语言:javascript
复制
public class MyClassLoader extends ClassLoader {
    public MyClassLoader() {    
        super();
    }

    @Override
    public synchronized Class<?> loadClass(String name) 
            throws ClassNotFoundException {

        System.out.println("Load: " + name);

        return super.loadClass(name);
    }
}

现在我尝试更改当前线程中的默认类加载器:

代码语言:javascript
复制
import java.util.ArrayList;
import java.util.List;

public class ChangeLoaderTest {
    public static void main(String[] args) {
        // Save class loader so that we can restore later.
        ClassLoader oldLoader = Thread.currentThread().getContextClassLoader();

        MyClassLoader newLoader = new MyClassLoader();
        try {
            // Set new classloader.
            Thread.currentThread().setContextClassLoader(newLoader);

            // My class.
            A a = new A("1");
            a.methodA();

            // Standard Java class.
            List<Integer> list = new ArrayList<Integer>();
            list.add(2);
            list.add(3);
        } finally {
            // Restore.
            Thread.currentThread().setContextClassLoader(oldLoader);
        }
    }
}

ChangeLoaderTest输出:

代码语言:javascript
复制
Create A instance: 1
#methodA1()

无名之辈

代码语言:javascript
复制
Load: ...

为什么?我怎样才能把ClassLoader变成一些线程?

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2012-04-17 22:23:01

正如马尔科·托波尼克指出的那样,context classloader is for use by frameworks。要自己使用类加载器,您必须调用loadClass("somepackage.A"),然后使用反射API创建A (Class.newInstance())的新实例。

你不能直接在源代码中使用A或它的方法,因为调用代码不知道A-它使用了不同的类加载器。可以由普通类加载器加载的A的接口或基类可以用来避免反射。

代码语言:javascript
复制
interface AIF{
        void someMethod();
 }
class A implements AIF{
      public void someMethod(){}
 }


public void test(){
     MyLoader loader = new MyLoader();
     Class cla = loader.loadClass("A");
     AIF a = (AIF) cla.newInstance();
     a.someMethod();

 }
票数 4
EN

Stack Overflow用户

发布于 2012-04-17 21:58:41

contextClassLoader机制是,而不是new等基本Java操作所使用的。只有这样,各种框架才能访问负责的上下文类加载器,并加载资源、类等。Java将始终使用加载正在执行的代码的类加载器。它是通过ChangeLoaderTest.class.getClassLoader()访问的--对此您无能为力。

票数 3
EN

Stack Overflow用户

发布于 2012-04-17 21:58:13

我认为发生的事情是你的应用程序的类加载器,也就是你的类加载器的“父”可以定位A并加载它。因此,您的类加载器将不会被搜索或用于加载A

老实说,我没有太多使用类加载器的经验,但如果你子类化了一个使用URL作为类路径的类加载器(这样它就可以定位类文件),而父类加载器不能加载它(不是类路径的一部分),那么就会使用你的自定义类加载器。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/10192453

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档