是javap缺少一些行号?

时间:2016-01-21 12:38:04

标签: java bytecode javap

如果我编译以下源代码:

1.  public class Example {
2.    public boolean foo(boolean a, boolean b, char ch) {
3.      if (a && Character.isLetter(ch)
4.            || b && Character.isDigit(ch)
5.            || !a && !b) {
6.        return true;
7.      }
8.      return false;
9.    }
10. }

使用Java-7(OpenJDK或Oracle)我得到以下LineNumberTable:

line 3: 0
line 6: 30
line 8: 32

然而,使用Java-8(OpenJDK或Oracle)编译相同的源代码我得到另一个LineNumberTable:

line 3: 0
line 4: 16
line 6: 30
line 8: 32

为什么Java-7返回的LineNumberTable不包含第4行和第5行?以及为什么Java-8返回的LineNumberTable不包含第5行?

谢谢,

1 个答案:

答案 0 :(得分:3)

嗯,没有确切说明,编制者应该包含多少提示。它总是在有用性和最终代码大小之间进行权衡。在最坏的情况下,每个单字节指令都有几个字节长度的信息。此外,某些语言结构可能会生成分布在不同字节代码位置的指令,而其他语言结构根本不会映射到字节代码指令。

之前javac实施的策略很简单。与Eclipse不同,它不会为表达式生成行号信息,而只为语句生成行号信息。因此,单个语句可能跨越多个源代码行,但其整个代码仅与语句的第一行相关联。

显然,当存在嵌入式方法调用时,此策略已更改为对此规则进行例外处理。因此,对于方法调用,将会有行号信息,但不会出现其他表达式。

您可以通过使用故意添加或删除嵌套调用的多行语句来验证这一点。例如,以下代码

static int test(int a, int b, int c, int d) {
    return
            a
           +b
           +c
           +d;
}

只会报告一个带有return关键字的行号。现在将代码更改为

static int test(int a, int b, int c, int d) {
    return
            a
           +b
           +c+test(a,b,c,d)
           +d;
}

并且行号表将有两个条目,一个是return关键字,另一个是嵌套方法调用。您可以将调用移动到另一行,甚至可以添加另一个调用,例如

static int test(int a, int b, int c, int d) {
    return
            a
           +b+test(a,b,c,d)
           +c
           +d+test(a,b,c,d);
}

并且行号表将支持我对javac新策略的猜测。