如何从 Java 的 JDI 中隐藏变量?

How to hide variables from Java's JDI?

我正在检测一些 类 并引入一些新的局部变量。现在,当用户在代码中放置断点并停止执行时,可以在 Intellij IDEA 的调试器中看到新引入的局部变量 window。我怎样才能隐藏它们?

更新:我将不得不以某种方式从检测代码中删除调试信息,但不确定该怎么做。

更新 2: 我正在使用 ASM 库进行检测。

 public void visitCode() {
       this.mv.visitLdcInsn(stringToPass);
       this.mv.visitMethodInsn(Opcodes.INVOKESTATIC, "MyAgentClass", "loadData", "(Ljava/lang/String;)LDataClass;", false);
       this.mv.visitVarInsn(Opcodes.ASTORE, this.getDataIndex());
}


public void visitMaxs(int maxStack, int maxLocals) {
     if (this.myStartLabel != null && this.myEndLabel != null) {
        this.mv.visitLocalVariable("__my__data__", "Ljava/lang/Object;", (String) null, this.myStartLabel, this.myEndLabel, this.getDataIndex());
     }

       super.visitMaxs(maxStack, maxLocals);
 }

__my__data__ 显示在 Intellij IDEA 中。

https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.7.13

根据 JVM 规范,您可以从 LocalVariableTable 中删除局部变量。 Javassist 在生成的代码中自动执行此操作,我在 运行:

期间看不到变量 secretCode

反编译结果class可以看出没有LocalVariableTable个条目:


  public void run();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=3, args_size=1
         0: ldc           #39                 // int -889275714
         2: istore_1
         3: iload_1
         4: invokestatic  #43                 // Method org/example/App.test:(I)Ljava/lang/Integer;
         7: astore_2
         8: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        11: aload_2
        12: invokevirtual #5                  // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
        15: new           #2                  // class SecreFoo
        18: dup
        19: invokespecial #3                  // Method "<init>":()V
        22: astore_1
        23: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        26: aload_1
        27: invokevirtual #5                  // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
        30: return
      LineNumberTable:
        line 14: 15
        line 15: 23
        line 16: 30
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      31     0  this   LSecreFoo;
           23       8     1 fCopy   LSecreFoo;

因此,当您检测 class 时,从 table 中删除所有局部变量(或者不添加它们)。