TypeToken获取运行时泛型类型

 

最近正好使用到了Guava的TypeToken来获取泛型的类型信息

 

比如,泛型父类需要获取其子类定义的泛型类型时:

public abstract class GenericClazz<V> {
  
  private Class<V> classType;
  
  public void doSth() {
    final TypeToken<V> typeToken = new TypeToken<V>(getClass()) {};
    classType = (Class<V>) typeToken.getRawType();  //获得子类的泛型类型
  }
}

 

而使用反射,就稍微复杂了一点。

public abstract class ReflectClazz<V> {
  
  private Class<V> classType;
  
  public void doSth() {
    final Type genType = getClass().getGenericSuperclass();
    classType = (Class<V>) ((ParameterizedType) genType).getActualTypeArguments()[0];  //获得子类的泛型类型
  }
}

而当继承类申明的泛型V也是个泛型类,如 public class SubClazz extends ReflectClazz<Map<Integer, String>> {} 这种,使用反射就会更加繁琐……

 

 

还有一种情况,当我们需要在方法/局部变量中获取泛型类型时,也可以使用TypeToken:

  public void getGenericType() {
    final TypeToken typeToken = new TypeToken<List<Integer>>() {};
    final Type type = typeToken.getType(); //java.util.List<java.lang.Integer>
  }

上面使用TypeToken的目的是为了在运行时获取List<T>的泛型类型Integer,反射则办不到。

(类型擦除机制,.class的LocalVariableTable属性中只会保留Ljava/util/ArrayList,不会是 Ljava/util/ArrayList<Ljava/lang/Integer;> )

 

TypeToken则不是直接利用反射,而是曲线救国:创建一个TypeToken<T>的匿名继承类。由于匿名类的申明信息中保留了泛型信息,通过反射可得…

(编译器不会擦除声明信息,Signatur属性中会保存该匿名类的完整信息,如 Lcom/google/common/reflect/TypeToken<Ljava/util/List<Ljava/lang/Integer;>;>; )。

 

posted @ 2018-01-15 00:08  傅晓芸  阅读(5657)  评论(1编辑  收藏  举报