
Java泛型在运行时被类型擦除,仅保留上界(无上界则为Object),导致List和List在JVM中均为List,Class对象相同,无法通过反射直接获取泛型参数。
Java的泛型是伪泛型,编译期存在,运行时被擦除。这意味着 List 和 List 在JVM里都是 List,底层Class对象完全相同。
擦除后,所有泛型参数统一替换为上界:没有显式上界(如 )则替换成 Object;有上界(如 )则替换成 Number。这也解释了为什么不能用 new T() 或 T.class —— 类型 T 在运行时已不存在。
getGenericXxx() 系列方法(如 Method.getGenericReturnType()),且仅对成员签名中的泛型有效,对局部变量或方法调用返回值无效new ArrayList[10] 编译报错,因为数组需要运行时类型信息,而泛型已被擦除static T value 是非法的ArrayList 和 ArrayList 不能构成重载因为擦除后二者都变成 ArrayList,方法签名在字节码层面完全一致,JVM无法区分。以下代码编译失败:
void foo(ArrayListlist) {} void foo(ArrayList list) {} // 编译错误:重复的方法签名
但可以和原始类型(raw type)重载成功,因为 ArrayList 和 ArrayList 擦除后虽然类型一致,但编译器仍视其为不同签名(原始类型不参与类型检查)。
int flag 参数)当必须在运行时知道泛型类型(如JSON反序列化、ORM映射),常见做法是传入一个匿名子类来捕获泛型信息:
new TypeReference>() {}
原理是:匿名子类会把父类的泛型签名写进字节码的 Signature 属性,通过 getClass().getGenericSuperclass() 可提取出来。但注意这仅适用于继承关系明确、泛型出现在父类声明中的场景。
List field 的 T
无法通过 field.getClass() 获取ParameterizedTypeReference、Jackson的 TypeReference 都基于同一机制Map>> ),手动构造 TypeReference 易出错,建议用 TypeFactory.constructXXX()(Jackson)或 ResolvableType.forInstance()(Spring)最常踩的坑是类型转换异常和集合误用。例如:
Liststrings = new ArrayList<>(); List raw = strings; raw.add(123); // 编译通过,但破坏了strings的类型契约 String s = strings.get(0); // ClassCastException: Integer cannot be cast to String
这种错误在混合使用泛型与原始类型时极易发生,且只在运行时暴露。
session.createCriteria(...).list()),强转成泛型集合前务必确认元素真实类型instanceof 不能用于参数化类型:if (obj instanceof ArrayList) 编译失败,只能写成 obj instanceof ArrayList
gson.fromJson(json, List.class) 返回的是 ArrayList,内部元素仍是 LinkedTreeMap,需配合 TypeToken 使用泛型擦除不是缺陷,而是Java为兼容老版本做的取舍。真正麻烦的不是擦除本身,而是开发者误以为“写了泛型就等于运行时安全”。只要记住:泛型校验止于编译期,运行时一切皆 Object。