前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >在Feign接口中返回泛型类型——自定义Decoder

在Feign接口中返回泛型类型——自定义Decoder

作者头像
DannyHoo
发布2020-06-28 11:15:07
8.1K0
发布2020-06-28 11:15:07
举报
文章被收录于专栏:Danny的专栏

前几天对接了一套第三方接口,这几个第三方接口的请求地址一样,请求参数和响应结果中有很多共同的字段,所以就想把这些字段都抽出来,通过Feign定义的接口返回类型直接返回泛型。

Feign定义

代码语言:javascript
复制
@FeignClient(name = "demoFeign", url = "${config.demo.domain}")
public interface DemoFeign {

    @PostMapping(value = "/open/post")
    public <R extends BaseResponse, T extends BaseRequest> R invoke(@RequestBody T request);
            
}

请求参数父类 BaseRequest

代码语言:javascript
复制
@Data
public class BaseRequest{
    private String requestId;
    private String timeStamp;
    private String method;
}

接口1的请求参数定义 Request01

代码语言:javascript
复制
@Data
public class Request01 extends BaseRequest{
    private String merchantId;
}

接口2的请求参数定义 Request02

代码语言:javascript
复制
@Data
public class Request02 extends BaseRequest{
    private String orderNo;
}

响应结果父类 BaseRequest

代码语言:javascript
复制
@Data
public class BaseResponse{
    private String code;
    private String message;
}

接口1的响应结果定义 Response01

代码语言:javascript
复制
@Data
public class Response01 extends BaseResponse{
    private String merchantId;
    private String merchantName;
}

接口2的响应结果定义 Response02

代码语言:javascript
复制
@Data
public class Response02 extends BaseResponse{
    private String orderNo;
    private String orderTime;
}

调用的时候报错:feign.codec.DecodeException: type is not an instance of Class or ParameterizedType: R

代码语言:javascript
复制
feign.codec.DecodeException: type is not an instance of Class or ParameterizedType: R
	at org.springframework.cloud.openfeign.support.SpringDecoder.decode(SpringDecoder.java:61)
	at org.springframework.cloud.openfeign.support.ResponseEntityDecoder.decode(ResponseEntityDecoder.java:62)
	at feign.optionals.OptionalDecoder.decode(OptionalDecoder.java:36)
	at feign.SynchronousMethodHandler.decode(SynchronousMethodHandler.java:176)
	at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java:140)
	at feign.SynchronousMethodHandler.invoke(SynchronousMethodHandler.java:78)
	at feign.ReflectiveFeign$FeignInvocationHandler.invoke(ReflectiveFeign.java:103)
	at com.sun.proxy.$Proxy129.invoke(Unknown Source)

原来是当接口返回类型定义成泛型时,Feign的解码器Decoder(Feign默认的解码器是SpringDecoder)在解析接口响应内容的时候,Type被解析成了TypeVariableImpl类型,导致反序列化响应内容失败。

在这里插入图片描述
在这里插入图片描述

Feign的编码器和解码器是可插拔的,可以自定义一个Feign的解码器来解决这个问题。

1、定义一个 解析 返回类型为泛型Feign接口解码器GenericsFeignResultDecoder,需要实现Decoder接口;

2、定义一个CustomizedConfiguration类,用于包装GenericsFeignResultDecoder实例,用configuration属性为Feign指定自当前配置类。

代码语言:javascript
复制
@FeignClient(name = "demoFeign", url = "${config.demo.domain}", configuration = DemoFeign.CustomizedConfiguration.class)
public interface DemoFeign {

    @PostMapping(value = "/open/post")
    public <R extends BaseResponse, T extends BaseRequest> R invoke(@RequestBody T request);
    
    public class CustomizedConfiguration{
        @Bean
        public Decoder feignDecoder() {
            return new FeignResultDecoder();
        }
    }

	public class GenericsFeignResultDecoder implements Decoder {
	    @Override
	    public Object decode(Response response, Type type) throws IOException, DecodeException, FeignException {
	        if (response.body() == null) {
	            throw new DecodeException(response.status(), "no data response");
	        }
	        Class returnType=((Method)((TypeVariableImpl)type).getGenericDeclaration()).getReturnType();
	        String bodyStr = Util.toString(response.body().asReader(Util.UTF_8));
	        return JSON.parseObject(bodyStr,returnType);
	    }
	}
}

如果要为当前Spring容器管理的所有Feign都指定这个解码器,就把CustomizedConfiguration类挪到Feign接口外面,再加@Configuration,我这里为了方便就写到Feign接口里了;如果只是为一个Feign Client指定自定义的解码器,GenericsFeignResultDecoder就不要加Spring注解(不要被Spring管理)了,否则就成了全局的了。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/06/23 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档