我试图在由另一个类加载器加载的类中引用通过类加载器加载的对象。我需要对该对象进行强制转换才能访问某些方法。我需要一种方法来避免其他类加载器进行此转换。 我知道我们无法转换不同类加载器的类,而这不是我在这里需要的。我需要在同一个类加载器中进行转换,以避免加载相同类的其他类加载器。结果真的很难。
这里有一些背景信息。正如您在下面的类中看到的,我通过 ComponentResolver.lookup() 方法在 MyClass.configure() 方法中收到一个对象,该对象是 Mojo 的实例,但转换为 Object。
不幸的是,Mojo mojo = clazz.cast(o)
(在下面的 MyClass 中)因编译错误而失败,表明 clazz.cast 的返回类型是 Object。谁能告诉我如何解决这个问题?可能通过相同的 Class#cast 方法或通过反射?不过我对反射不太熟悉!
//This is loaded by, say ClassLoader X
public class ComponentResolver {
public Object lookup(String role) {
//do something
return component; //component is an instance of Mojo interface.
}
}
这是我调用查找方法的 MyClass。
//This class including Mojo in this context is loaded by, say ClassLoader Y
public class MyClass {
public void configure() {
Object o = componentResolver.lookup("componentName");
// Mojo mojo = (Mojo) o; //causes classcastexception (obviously.)
Class<?> clazz = Class.forName("org.Mojo", false,
o.getClassLoader() );
Mojo mojo = clazz.cast(o);
//Causes compiler errors because the returned object is of type Object.
//ie incompatible types Required:Mojo, Found:Object
// Mojo mojo = (Mojo) clazz.cast(o); //again classcastexception.
mojo.execute();
}
}
最佳答案
就 JVM 而言,您的类加载器加载的 Mojo
类和其他类加载器加载的 Mojo
类是完全不同的类。
一些选项:
- 通过反射调用方法等,做任何您需要做的事情
- 通过反射基于现有的
Mojo
创建一个新的Mojo
,然后使用它(这取决于Mojo
真正在做什么) - 更改您的类加载器层次结构,以便将一个类加载器委托(delegate)给另一个类加载器,以便您可以进行强制转换。
您的处境从根本上来说很糟糕,而且据我所知,没有简单的解决方法。如果您可能可以修复类加载器层次结构,那么您的生活会变得更加轻松。只需在类加载器中包含 Mojo
,其他两个类加载器都将其作为其父类加载器就足够了。
关于java - 两个类加载器;一个类加载器需要强制转换一个对象,但其他类加载器不允许这样做,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7130667/