java - 合成方法的惩罚是什么?

标签 java jvm java-synthetic-methods

在 Eclipse 下开发 Java 应用程序时,我收到有关“通过合成方法访问的方法/值”的警告。解决方案只是将私有(private)访问修饰符更改为默认级别。

这让我想知道:使用合成方法的代价是什么?有一些?我假设是这样,因为编译器/Eclipse 会发出警告,但它是如此相关或可以安全地忽略的东西?

我在这里没有看到这方面的信息,所以我问一下。

最佳答案

Eclipse 警告您可能会泄露您认为是隐私的信息。合成访问器可以被恶意代码利用,如下所示。

如果您的代码需要在安全的 VM 中运行,使用内部类可能是不明智的。如果您可以使用反射并且可以完全访问所有内容,那么合成访问器不太可能产生可衡量的差异。


例如,考虑这个类:

public class Foo {
  private Object baz = "Hello";
  private class Bar {
    private Bar() {
      System.out.println(baz);
    }
  }
}

Foo 的签名实际上是:

public class Foo extends java.lang.Object{
    public Foo();
    static java.lang.Object access$000(Foo);
}

access$000 是自动生成的,让单独的类 Bar 访问 baz 并且会被标记为 Synthetic属性。生成的精确名称取决于实现。常规编译器不会让您针对此方法进行编译,但您可以像这样使用 ASM(或类似的)生成您自己的类:

import org.objectweb.asm.*;
public class FooSpyMaker implements Opcodes {
  public static byte[] dump() throws Exception {
    ClassWriter cw = new ClassWriter(0);
    cw.visit(V1_6, ACC_PUBLIC + ACC_SUPER, "Spy", null, "java/lang/Object",null);
    MethodVisitor ctor = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
    ctor.visitCode();
    ctor.visitVarInsn(ALOAD, 0);
    ctor.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V");
    ctor.visitInsn(RETURN);
    ctor.visitMaxs(1, 1);
    ctor.visitEnd();
    MethodVisitor getBaz = cw.visitMethod(ACC_PUBLIC, "getBaz",
        "(LFoo;)Ljava/lang/Object;", null, null);
    getBaz.visitCode();
    getBaz.visitVarInsn(ALOAD, 1);
    getBaz.visitMethodInsn(INVOKESTATIC, "Foo", "access$000",
        "(LFoo;)Ljava/lang/Object;");
    getBaz.visitInsn(ARETURN);
    getBaz.visitMaxs(1, 2);
    getBaz.visitEnd();
    cw.visitEnd();
    return cw.toByteArray();
  }
}

这将创建一个名为 Spy 的简单类,它允许您调用 access$000:

public class Spy extends java.lang.Object{
    public Spy();
    public java.lang.Object getBaz(Foo);
}

使用它,您可以检查 baz 的值,而无需反射或任何暴露它的方法。

public class Test {
  public static void main(String[] args) {
    Foo foo = new Foo();
    Spy spy = new Spy();
    System.out.println(spy.getBaz(foo));
  }
}

Spy 实现要求它与 Foo 在同一个包中,并且 Foo 不在 sealed JAR 中。 .

关于java - 合成方法的惩罚是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5557955/

相关文章:

java - 如何返回已在 4 个不同线程中编辑过的变量的一个实例?

java - 使用列表删除数组

java - 可见范围项目/模块

java - 综合访问器方法警告

Java合成方法与桥接方法混淆

java - 如何判断提交的表单是否有多个同名元素

java - 如何使用 BigInteger 修复 Polynomial 类中的这个 "plus"方法

java - 是否有用于更改目录的 JVM 命令行选项?

mono - 在 CLR 和 JVM 中如何实现 Object.GetHashCode()?

java - 如何使用内联匿名类声明避免合成访问编译器警告,这是什么意思?