Java泛型类型擦除问题
以前就了解过Java泛型的实现是不完整的,最近在做一些代码重构的时候遇到一些Java泛型类型擦除的问题,简单的来说,Java泛型中所指定的类型在编译时会将其去除,因此List<String>
和 List
在编译成字节码的时候实际上是一样的。因此java泛型只能做到编译期检查的功能,运行期间就不能保证类型安全。我最近遇到的一个问题如下:
假设有两个bean类
/** Test. */ @Data @NoArgsConstructor @AllArgsConstructor public static class Foo { public String name; } /** Test. */ @Data @NoArgsConstructor @AllArgsConstructor public static class Dummy { public String name; }
以及另一个对象
@NoArgsConstructor @AllArgsConstructor @Data public static class Spec<T> { public String spec; public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); return (T) mapper.readValue(spec, Foo.class); } }
可以看到Spec
对象中保存了以上两种类型json序列化后的字符串,并提供了方法将string spec 反序列化成相应的类型,比较理想的方式是在反序列化的方法中能够获取到参数类型 T 的实际类型,理论上运行时Spec类型是确定了,因此T也应该是确定的,但是因为类型擦除,所以实际上获取不到他的类型。
按照以下尝试 通过((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()
获取泛型类型,经过测试是获取不到的
@Test public void test() throws JsonProcessingException { var foo = new Foo("foo"); var spec = new Spec<Foo>(mapper.writeValueAsString(foo)); var deserialized = spec.deserializeTo(); Assertions.assertTrue(deserialized instanceof Foo); } @NoArgsConstructor @AllArgsConstructor @Data public static class Spec<T> { public String spec; private Class<T> getSpecClass() { return (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()) .getActualTypeArguments()[0]; } public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); System.out.println(spec); return (T) mapper.readValue(spec, getSpecClass()); } }
会有以下的错误
java.lang.ClassCastException: class java.lang.Class cannot be cast to class java.lang.reflect.ParameterizedType (java.lang.Class and java.lang.reflect.ParameterizedType are in module java.base of loader 'bootstrap')
有两种办法来绕过这个问题
第一种比较简单,就是在创建spec对象时,直接把类型的class传进来,这样就可以直接使用。
第二种是创建spec的子类中使用这个方法就可以获取泛型的类型
@Data public abstract static class AbstractSpec<T> { public String spec; public AbstractSpec(String spec) { this.spec = spec; } private Class<T> getSpecClass() { return (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()) .getActualTypeArguments()[0]; } public T deserializeTo() throws JsonProcessingException { var mapper = new ObjectMapper(); System.out.println(spec); return (T) mapper.readValue(spec, getSpecClass()); } } public static class Spec extends AbstractSpec<Foo> { public Spec(String spec) { super(spec); } } @Test public void test() throws JsonProcessingException { var foo = new Foo("foo"); var spec = new Spec(mapper.writeValueAsString(foo)); var deserialized = spec.deserializeTo(); Assertions.assertTrue(deserialized instanceof Foo); }
这里spec类就可以顺利的被反序列化。
这个和最开始失败的case的差别就是新增了一个子类,主要的差别是getGenericSuperclass的返回值有差异,非子类的情况下,获取到的是Object。
因此理论上子类Spec的类型信息中,实际上是保存了父类中的类型参数信息的,也就是例子中的Foo. 按照 https://stackoverflow.com/questions/42874197/getgenericsuperclass-in-java-how-does-it-work 的方式,可以查看到Spec类的字节码中有相应的类型信息。
$ javap -verbose ./org/apache/flink/kubernetes/operator/controller/GenericTest\$Spec.class | grep Signature #15 = Utf8 Signature Start Length Slot Name Signature Signature: #19 // Lorg/apache/flink/kubernetes/operator/controller/GenericTest$AbstractSpec<Lorg/apache/flink/kubernetes/operator/controller/GenericTest$Foo;>;
参考
https://www.cnblogs.com/wuqinglong/p/9456193.html
https://stackoverflow.com/questions/3403909/get-generic-type-of-class-at-runtime
https://stackoverflow.com/questions/6624113/get-type-name-for-generic-parameter-of-generic-class
https://github.com/jhalterman/typetools
本文来自博客园,作者:Aitozi,转载请注明原文链接:https://www.cnblogs.com/Aitozi/p/16280684.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!