跳过Java初始化字节码

dav*_*404 3 java bytecode

在类文件的字节码中javap -c ...,有一个代码块负责初始化类中的静态字段.但是,有两个if语句,ifne并且ifeq相互放置:

27: ifne 120
30: ifeq 119
Run Code Online (Sandbox Code Playgroud)

初始化块位于偏移量37和115之间,并且是初始化类中包含的静态变量所必需的.我的理解是,ifne如果堆栈上的值不为0,ifeq则会进行分支,如果堆栈上的值为0,则会跳转,这意味着它将分支.

我的问题是:如果初始化块只是被这些分支跳过,为什么变量稍后会被初始化?我找不到任何分支到块,所以如果代码在其他地方分支,它怎么能运行?

编辑:更多字节码

0: getstatic      #602  // static integer field
3: istore         25
...
25: iload         25
27: ifne          120
30: ifeq          119
33: goto          37
36: athrow
37: new           #6    // class java/io/File
40: dup
41: getstatic     #615  // static string containing filename
...
56: invokespecial #9    // Method java/io/File."<init>":(Ljava/lang/String;)V
...
112: putstatic    #14   // static File referenced later on
Run Code Online (Sandbox Code Playgroud)

cod*_*eim 6

Java不是我最强大的VM,但我通常理解操作码.

27: ifne 120
30: ifeq 119
Run Code Online (Sandbox Code Playgroud)

条件分支从堆栈中弹出最高值.所以这不是无条件分支,它是基于2个不同(连续)堆栈值的2个分支指令.简而言之,它消耗了2个不同的值.

因此,如果堆栈上有0和1,它将在没有分支的情况下掉落.

此外,静态字段初始化不能保证立即执行,我所拥有的JVM规范(旧的)说它会在第一次非常规访问类时延迟执行.这可能与您的问题无关,但值得一提.

最后但并非最不重要的是,并非所有字节码都有效.混淆器将根据定义插入无法访问,冗余或"混淆"的无效/死代码.

  • 是的:"值必须是int类型.它从操作数堆栈中弹出并与零进行比较." (2认同)