0

下面我粘贴了我想运行的示例。

我可以很容易地得到一个StringHome实例的泛型类 extends Home<String>,但我不能对new Home<String>()实例做同样的事情。我怎样才能获得它?

我的目标是使mainKO方法正常工作mainOK

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;

public class Home<E> {
    @SuppressWarnings ("unchecked")
    public Class<E> getTypeParameterClass(){
        Type type = getClass().getGenericSuperclass();
        ParameterizedType paramType = (ParameterizedType) type;
        return (Class<E>) paramType.getActualTypeArguments()[0];
    }

    private static class StringHome extends Home<String>{}
    private static class StringBuilderHome extends Home<StringBuilder>{}
    private static class StringBufferHome extends Home<StringBuffer>{}   

    public static void main(String[] args) throws Exception {
        // this works fine
        mainOK();
        Thread.sleep(1000);
        // this throws an error
        mainKO();
    }

    /**
     * This prints "String", "StringBuilder" and "StringBuffer"
     */
    public static void mainOK() throws Exception {
        Object object0 = new StringHome().getTypeParameterClass().newInstance();
        Object object1 = new StringBuilderHome().getTypeParameterClass().newInstance();
        Object object2 = new StringBufferHome().getTypeParameterClass().newInstance();
        System.out.println(object0.getClass().getSimpleName());
        System.out.println(object1.getClass().getSimpleName());
        System.out.println(object2.getClass().getSimpleName());
    }

    /**
     * This throws error:
     * Exception in thread "main" java.lang.ClassCastException: java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType
     */
    public static void mainKO() throws Exception {
        Object object0 = new Home<String>().getTypeParameterClass().newInstance();
        Object object1 = new Home<StringBuilder>().getTypeParameterClass().newInstance();
        Object object2 = new Home<StringBuffer>().getTypeParameterClass().newInstance();
        System.out.println(object0.getClass().getSimpleName());
        System.out.println(object1.getClass().getSimpleName());
        System.out.println(object2.getClass().getSimpleName());
    }
}

这个类的执行打印出这个:

String
StringBuilder
StringBuffer
Exception in thread "main" java.lang.ClassCastException: java.lang.Class cannot be cast to java.lang.reflect.ParameterizedType
    at org.command4j.core.commands.utils.Home.getTypeParameterClass(Home.java:13)
    at org.command4j.core.commands.utils.Home.mainKO(Home.java:46)
    at org.command4j.core.commands.utils.Home.main(Home.java:26)
4

1 回答 1

2

你不能。

要理解这一点,只需为每个对象的层次结构制作一棵树。

  • StringHome的超类是Home<java.lang.String>。因此,当您调用 getGenericSuperclass() 时,您将获得该超类。然后你可以得到实际的类型参数String

  • Home<String>的超类是java.lang.Object。此外,在运行时,您丢失了参数并且只知道它是一个Home对象。

Java 泛型在编译时会擦除泛型类型信息。您将无法在运行时获取此信息。

于 2016-07-01T15:15:11.603 回答