我有一些A类
public class A {
public A(String str) {
System.out.println("Create A instance: " + str);
}
public void methodA() {
System.out.println("#methodA1()");
}
}和我的类加载器实现:
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);
}
}现在我尝试更改当前线程中的默认类加载器:
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输出:
Create A instance: 1
#methodA1()无名之辈
Load: ...为什么?我怎样才能把ClassLoader变成一些线程?
发布于 2012-04-17 22:23:01
正如马尔科·托波尼克指出的那样,context classloader is for use by frameworks。要自己使用类加载器,您必须调用loadClass("somepackage.A"),然后使用反射API创建A (Class.newInstance())的新实例。
你不能直接在源代码中使用A或它的方法,因为调用代码不知道A-它使用了不同的类加载器。可以由普通类加载器加载的A的接口或基类可以用来避免反射。
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();
}发布于 2012-04-17 21:58:41
contextClassLoader机制是,而不是new等基本Java操作所使用的。只有这样,各种框架才能访问负责的上下文类加载器,并加载资源、类等。Java将始终使用加载正在执行的代码的类加载器。它是通过ChangeLoaderTest.class.getClassLoader()访问的--对此您无能为力。
发布于 2012-04-17 21:58:13
我认为发生的事情是你的应用程序的类加载器,也就是你的类加载器的“父”可以定位A并加载它。因此,您的类加载器将不会被搜索或用于加载A。
老实说,我没有太多使用类加载器的经验,但如果你子类化了一个使用URL作为类路径的类加载器(这样它就可以定位类文件),而父类加载器不能加载它(不是类路径的一部分),那么就会使用你的自定义类加载器。
https://stackoverflow.com/questions/10192453
复制相似问题