我对类加载的理解是,一个类在第一次需要时被加载(用一种非常简单的方式说)。使用 -verbose:class 和 Iterators 类的修改版本运行以下示例,该版本在调用其 clinit 时会打印一条消息,但我观察到了一些我无法真正解释的东西:
public class IteratorsTest
{
public static void main(String[] args)
{
com.google.common.collect.Iterators.forArray(1, 2, 3);
}
}
(清理后的)输出如下:
[Loaded com.google.common.collect.Iterators from file:...]
[Loaded com.google.common.collect.Iterators$1 from file:...]
---------> Iterators <clinit>
为什么在调用 clinit 之前加载 Iterators$1?它只在 clinit 中定义,不是吗?
static final UnmodifiableListIterator<Object> EMPTY_LIST_ITERATOR =
new UnmodifiableListIterator<Object>() {
...
}
这导致以下字节码:
static <clinit>()V
L0
GETSTATIC java/lang/System.out : Ljava/io/PrintStream;
LDC "---------> Iterators clinit --------------"**
INVOKEVIRTUAL java/io/PrintStream.println (Ljava/lang/String;)V
L1
NEW com/google/common/collect/Iterators$1
DUP
INVOKESPECIAL com/google/common/collect/Iterators$1.<init> ()V
L2
PUTSTATIC com/google/common/collect/Iterators.EMPTY_LIST_ITERATOR : Lcom/google/common/collect/UnmodifiableListIterator;
更让我困惑的是,我还有一个示例(太复杂,无法在此处发布),其中与上面主要代码相同的代码行导致以下输出:
[Loaded com.google.common.collect.Iterators from file:...]
---------> Iterators <clinit>
[Loaded com.google.common.collect.Iterators$1 from file:...]
这实际上也是我对简单测试程序的期望。
我试图在这里找到答案https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-5.html,但这并没有真正帮助。
- 有时首先执行 clinit 而有时首先加载匿名类可能是什么原因?
- 当JVM调用类的clinit时,有没有办法跟踪?类似于 -verbose:class 或 -XX:+TraceClassLoading 等的东西?