[英]which classloader uses for composition classes
分析以下代码:
public class A {
public void foo() {
B b = new B(); // row#1
b.toString();
}
据我了解,在row#1
,类是由A
类的类加载器加载的。
考虑B
类:
class B {
C c = new C();
}
请阐明将使用哪个类加载器来加载类C
在获得类B
的类加载器并编写以下代码之后:
classloaderOfClassB.loadClass("A")
请说明所有涉及的类将使用什么类加载器。
我用以下示例产生以下输出。
不知道类C
,我假设它位于A
内。 如果不是,此stacktrace应该可以为您提供帮助。
AFAIK, A
加载C
产量
java.lang.Thread.getStackTrace(Thread.java:1568)
A$C.<init>(Main.java:15)
<-在这里
A$B.<init>(Main.java:10)
A.foo(Main.java:5)
A.main(Main.java:23)
例
import java.lang.StackTraceElement;
class A {
public void foo() {
B b = new B(); // row#1
b.toString();
}
class B {
C c = new C();
}
class C {
public C() {
for (StackTraceElement ele : Thread.currentThread().getStackTrace()) {
System.out.println(ele);
}
}
}
public static void main (String[] args) {
A a = new A();
a.foo();
}
}
如果一个类依赖于其他类,例如在您的示例中,类A依赖于类B,而类B本身依赖于类C,另外这些类尚未加载,则当前类的类加载器将用于尝试加载从属类。类。
实际上,以您的示例为例,类A的类加载器也将用于加载类B和C。
该声明
classloaderOfClassB.loadClass("A")
现在将加载类A并将类加载器设置为classloaderOfClassB
引用的实例。 然后,将使用相同的类加载器加载类A中所有其他尚未加载的依赖项。并且由于类B已由相同的加载器加载,因此一切都很好。
此机制确保在以下情况下类加载正常进行:
考虑有一个可插入的应用程序,其中插入意味着将JAR文件(不是类路径的一部分)放入指定的目录中。 然后,您可以使用由JAR文件构造的URLClassLoader
(当然是URL形式)来加载此新JAR的类和资源。 由于所有类现在都具有此URLClassLoader
作为其类加载器,因此所有进一步的依赖项将由相同的URLClassLoader
加载,这是正确的,因为只有此URLClassLoader
此JAR文件。
OSGi将使用类B的类加载器加载类C。因此,例如,如果B和C驻留在同一个包中,但是C在内部包中,则它仍然可以工作。 原因是,当类A的类加载器加载类B时,它将委托给B所在的捆绑软件的类加载器。 因此,B将使用其所在的捆绑包的类加载器加载。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.