RestTemplate将响应数据转换为具有泛型的类对象

一:没有泛型的情况(这里以POST举例,但是对于GET也是一样的)

CommonResponse response = restTemplate.postForObject(remoteUrl, tokenRequest, CommonResponse.class);

这里tokenRequest是请求体对象,将会由Fastjson序列化为JSON字符串,它所属的类无所谓是不是有泛型,因为可以通过member.getClass()获得(即tokenRequest中所有的成员的类型是确立的且可获取的);

二:有泛型的情况

CommonResponse<TokenDataModel> response = restTemplate.postForObject(remoteUrl, tokenRequest, CommonResponse.class);

这个时候就有问题了,因为CommonResponse在字节码里是泛型擦除的,它所有的泛型成员本质上就是一个Object类型的子类但是.class里没有记录到底是什么具体类,因此当获取的response的JSON字符串中:

{
    "code": 8209,
    "code_description": "APPLICANT_SUCCESS",
    "data": {
        "cell_phone_number": "1111111111",
        "collect_website": "中国联通","token": "77785fb276ed47cb9a01ec52f9a133c8"
    },
    "message": "成功"
}

将data部分转换为泛型成员时RestTemplate将不知道该怎么转换,如果将T类型认为是Object,那么显然Object是没有cellPhoneNumber之类的属性的,所以RestTemplate的做法是将data部分的JSON字符串转换为了LinkedHashMap对象(key-value对);

三:解决方式

通过RestTemplate的exchange方法来解决:

ParameterizedTypeReference<CommonResponse<TokenDataModel>> typeRef = new ParameterizedTypeReference<CommonResponse<TokenDataModel>>() {};
// exchange并不对应于任何一种method,如GET、POST、PUT、DELETE,因此需要手动指定 CommonResponse
<TokenDataModel> response = restTemplate.exchange(remoteUrl, HttpMethod.POST, new HttpEntity<>(tokenRequest), typeRef).getBody();

四:单纯用Fastjson转换有泛型的类的方式

CommonResponse<TokenDataModel> response = JSON.parseObject("{json string}", new com.alibaba.fastjson.TypeReference<CommonResponse<TokenDataModel>>(){});

五:原理

在java里,泛型类编译成.class后确实是泛型擦除,但是泛型类对象里是会(间接)存储泛型的实际类型的,因此需要new TypeReference<Test<RealType>>(){}产生一个对象;

它内部是:

// 获取匿名类对象的Generic父类,即Test<RealType>类型
// 注意superClass并不是Class类对象,它应该理解为:
// 为泛型类的子匿名类对象产生的一个与之一一对应的Type类对象(这个倒像真泛型)
// 这个Type类对象里存储了此匿名类对象的所有泛型信息
Type superClass = getClass().getGenericSuperclass();

猜你喜欢

转载自www.cnblogs.com/silentdoer/p/9012599.html
今日推荐