为什么 Java 不允许 List<ClassName>.class 这样的操作?

2018-10-18 16:40:10 +08:00
 abcbuzhiming
我知道有泛型擦除,但是我不知道该如何用泛型擦除来解释这个现象

在被一个序列化工具坑过后,经过测试,发现了一个我以前没注意到的 Java 特性盲点:

Java 允许这样操作:
List.class
但是他不允许这样的操作:
List<ClassName>.class
所以如果你有一个泛型方法需要传入 List<ClassName>的 Class<T>类型的时候,用如下方式是错的
Jackson2JsonRedisSerializer<List<ClassName>> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<List<ClassName>>(List<ClassName>.class);

只能用如下丑陋的方式
Jackson2JsonRedisSerializer<List> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<List.class>(List.class);
于是,你想限定 List 容器里装什么,对不起,你限制不了,而且这玩意还会给你个警告,根据放狗的结果,不上压制警告注解的注解没法解决

然后我又够了好久,才发现还有一个迂回战术,能让你对 List 容器内部的类型进行限制,写法是下面这样的
Jackson2JsonRedisSerializer<List<CustomerInfoDto>> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<List<CustomerInfoDto>>(
(Class<List<CustomerInfoDto>>) new ArrayList<CustomerInfoDto>().getClass());
新建一个容器的实现类泛型容器,然后调它的 getClass 方法。。。
虽然它也有一个必须用注解压制的警告,但是好歹能工作啊。

我自己查阅了不是关于 Java class 属性的资料,只能发现这玩意是编译后就被定义的属性,来自虚拟机本身,但是没有解释为什么 List<ClassName>.class 是不允许的。
所以,我的问题就是如何解释这个现象,其次就是如果我要拿到类似 List<ClassName>这样的类型的 Class<T>;只能像上面那种迂回手段那样建哥对象让后 getClass 吗?
11053 次点击
所在节点    Java
26 条回复
ddup
2018-10-18 21:34:56 +08:00
没办法,因为运行时里没有泛型,缺少必要的类型信息,只能如此。
反正既然是 Java 就不要介意美丑的问题。
leeg810312
2018-10-18 23:37:06 +08:00
Java 早期设计偷懒埋下的巨坑,一味要求兼容性,以致于无法做到运行时的泛型支持,没有办法反射获取泛型类型信息,一直到现在 Java11 了还是没有解决,哪有永久的兼容性啊?
deming
2018-10-19 09:58:29 +08:00
可以用这个方法:

class CustomerInfoList extends List<CustomerInfoDto> {};

然后就可以使用:

Jackson2JsonRedisSerializer<CustomerInfoList> serializer = ... 来实现。
q397064399
2018-10-19 10:18:30 +08:00
List<List<List<List>>>> 在 Java 看来就是 List<Object> 就对了 :doge 这个算是历史遗留了
vincenteof
2018-10-19 10:57:17 +08:00
运行时和编译期的区别啊,编译后并没有你写的那个东西
xuanbg
2018-10-20 18:33:14 +08:00
3 楼正解

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/498842

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX