我们如何看待编译器为Java中的类型擦除生成的字节码

时间:2012-07-16 10:04:53

标签: java type-erasure

  

可能重复:
  How to view Java's byte code?

我想知道在类型擦除之后是否可以在类文件中看到java编译器完成的代码更改。

我尝试过javap,但它使用参数化类型重建原始源文件。

教程(http://docs.oracle.com/javase/tutorial/java/generics/genTypes.html)讨论在类型擦除后用有界类型或对象替换参数化类型。我想看看编译器做了哪些更改,javap或任何反编译器都看不到。

的问候, sathwik

2 个答案:

答案 0 :(得分:1)

类型擦除字面意思是从结果字节代码中删除泛型类型。值得注意的是缺席。因此,您只能看到字节代码中没有它。

使用javap时,您会看到一些通用类型信息可用于类,方法和字段签名。这包括在内,因此编译器可以使用这些类来编译代码。它们在运行时不做任何事情,虽然它们可以通过反射获得,因此您可以编写一个使用此信息的库。

答案 1 :(得分:0)

我遇到的一个案例是Collections.max(),他们希望类型删除为Object,但仍然强制它为Comparable。你可以看到这样的签名:

vlad@vld /tmp $ javap java.util.Collections | grep max
    public static java.lang.Object max(java.util.Collection);
    public static java.lang.Object max(java.util.Collection, java.util.Comparator);

另一种方法是使用javap -c并查看编译器执行的检查。例如:

对于这个程序:

public class Test{
    public static void main(String[]args){
        java.util.Vector<String> a = new java.util.Vector<String>();
        a.add("Hello world");
        System.out.println(a.get(0).length());
    }
}

您会得到以下结果:

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

public static void main(java.lang.String[]);
  Code:
   0:    new    #2; //class java/util/Vector
   3:    dup
   4:    invokespecial    #3; //Method java/util/Vector."<init>":()V
   7:    astore_1
   8:    aload_1
   9:    ldc    #4; //String Hello world
   11:    invokevirtual    #5; //Method java/util/Vector.add:(Ljava/lang/Object;)Z
   14:    pop
   15:    getstatic    #6; //Field java/lang/System.out:Ljava/io/PrintStream;
   18:    aload_1
   19:    iconst_0
   20:    invokevirtual    #7; //Method java/util/Vector.get:(I)Ljava/lang/Object;
   23:    checkcast    #8; //class java/lang/String
   26:    invokevirtual    #9; //Method java/lang/String.length:()I
   29:    invokevirtual    #10; //Method java/io/PrintStream.println:(I)V
   32:    return

}

注意checkcast