4

每次我尝试理解编译后的 Java 文件的反汇编代码时,我都想知道为什么缺少一些指令号。

一个小例子:

我反汇编 ( $ javap -c HelloWorld) 一个简单的 HelloWorld 类。这是输出:

Compiled from "HelloWorld.java"
public class HelloWorld {
  public HelloWorld();
    Code:
       0: aload_0       
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return        

  public static void main(java.lang.String[]);
    Code:
       0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
       3: ldc           #3                  // String Hello World!
       5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
       8: return        
}

如您所见,构造函数中的指令 3 和 4 以及 main 方法中的一些指令缺失。

有人知道为什么缺少这些指令编号吗?是否有一些由于某种原因隐藏的字节码指令?

4

2 回答 2

9

“漏洞”是当前指令的参数所在的位置,请参阅Java 虚拟机规范。它包含第 6 章中的完整字节码列表。

例如invokevirtualinvokespecial两者都采用 2 个参数,因此下一个操作码将在 3 个字节后找到。在这两种情况下,都需要参数(indexbyte1 和 indexbyte2)来计算常量池中的位置:(indexbyte1 << 8) | indexbyte2)

Javap 会为您查找这些值,这是实际指令后注释中的引用。

于 2014-03-21T16:38:22.870 回答
0

这些不是指令编号,它们是指令相对于方法的字节偏移量。

为此,我仍在寻找官方文档,但此链接证实了这一点。

于 2014-03-21T16:38:30.433 回答