前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java泛型类型擦除问题

Java泛型类型擦除问题

作者头像
用户7353950
发布2022-06-23 15:55:45
1.1K0
发布2022-06-23 15:55:45
举报
文章被收录于专栏:IT技术订阅

Java泛型类型擦除问题

以前就了解过Java泛型的实现是不完整的,最近在做一些代码重构的时候遇到一些Java泛型类型擦除的问题,简单的来说,Java泛型中所指定的类型在编译时会将其去除,因此List<String>List 在编译成字节码的时候实际上是一样的。因此java泛型只能做到编译期检查的功能,运行期间就不能保证类型安全。我最近遇到的一个问题如下:

假设有两个bean类

代码语言:javascript
复制

/** Test. */
@Data
@NoArgsConstructor
@AllArgsConstructor
public static class Foo {
public String name;
}


/** Test. */
@Data
@NoArgsConstructor
@AllArgsConstructor
public static class Dummy {
public String name;
}

以及另一个对象

代码语言:javascript
复制

@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()获取泛型类型,经过测试是获取不到的

代码语言:javascript
复制

@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的子类中使用这个方法就可以获取泛型的类型

代码语言:javascript
复制

@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类的字节码中有相应的类型信息。

代码语言:javascript
复制


本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-05-27,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 IT技术订阅 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Java泛型类型擦除问题
相关产品与服务
文件存储
文件存储(Cloud File Storage,CFS)为您提供安全可靠、可扩展的共享文件存储服务。文件存储可与腾讯云服务器、容器服务、批量计算等服务搭配使用,为多个计算节点提供容量和性能可弹性扩展的高性能共享存储。腾讯云文件存储的管理界面简单、易使用,可实现对现有应用的无缝集成;按实际用量付费,为您节约成本,简化 IT 运维工作。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档