好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

Feign接口方法返回值设置方式

一、介绍

随着微服务的广泛应用,越来越多的企业都会使用微服务进行项目开发,在各个服务之间需要通过feign来进行通信,所以在feign调用 接口 中方法会接受其他服务接口不同类型 返回值 。

二、返回值设置

1、依据被调用服务接口设置相同返回类型

介绍:微服务A接口getUser 返回List<User>类型,微服务B通过feign调用方法也返回相同的结果类型。

特点:返回类型一一对应,在调用时不需要进行转化直接拿来就可以用。

缺点:扩展性不好,维护性不加。

解释:在目前springboot开发中,接口一般都会返回json类型数据(也就是使用@restController或者使用@ResponseBody注解修饰),就算是对象或者对象集合也是一样的,或者其他自己封装的返回对象。如果有许多不同的返回对象,当这些返回对象在A服务做了修改相应的在B服务的feign接口处也要做修改,相当麻烦且不符合java面向接口编程思想。

2、全部设置为String

介绍:微服务A接口只要被@restController或者使用@ResponseBody注解修饰我统统在微服务Bfeign接口使用String来进行接受。

特点:通过String来接受返回参数,格式一致。

缺点:后面调用需要转换。

代码介绍:

// 微服务A controller 已经使用@restController修饰 @PostMapping ( value = "/getAllQuestionBank" , produces = MediaType . APPLICATION_JSON_UTF8_VALUE ) public List < QuestionBankDto > getAllQuestionBank (){ return baseinfoQuestionMange . getAllQuestionBank (); }   // 微服务B feign接口 使用String进行接受 @PostMapping ( value = "/baseinfo/getAllQuestionBank" ) String getAllQuestionBank ();   // 微服务B 当需要调用feign信息时需要强转成List<QuestionBankDto>格式   // 调用微服务获取题库章节信息,该处强制使用try catch进行包裹 String questionBankInfo = baseInfoApi . getAllQuestionBank (); List < QuestionBankDto > mysqlQuestionBank = objectMapper . readValue ( questionBankInfo , new TypeReference < ArrayList < QuestionBankDto >>(){});

只要你导入一下springboot依赖,它默认就给你导入了Jackson jar包

注意事项:使用该方式时需要将ObjectMapper配置到bean容器中。

3、总结

项目不同,需求不同,两种方式没有谁对谁错之分,依据自己项目需求进行选择。

Feign 使用这几天遇到的一些问题

事情的起因要从我打算调用下paas模块服务开始

Feign 现在教程很多,然后引入下pom

<dependency> <groupId> org.springframework.cloud </groupId> <artifactId> spring-cloud-starter-openfeign </artifactId> </dependency>

自己定义下接口

@FeignClient ( name = "weixinTokenClient" , url = "https://qyapi.weixin.qq.com/cgi-bin" ) public interface IWeiXinTokenClient { /** * 获取应用的token * @param corpid * @param corpsecret * @return */ @RequestMapping ( value = "/gettoken" , method = RequestMethod . GET ) WeiXinTokenResultModel getToken ( @RequestParam String corpid , @RequestParam String corpsecret ); }

写个test类测试下,完美返回,因为自己写过类似的项目,可以看下lemur-http,原理大体一致,想了想还是简单的.但是后面的问题就出来

1. nacos 配置拿不到服务

原因是:nacos注册服务只注册了lemur-admin和lemur-paas这种服务级别的服务,获取服务地址需要用lemur-admin服务,但是在admin调用paas接口的时候

@FeignClient ( value = "paasUserFacade" , contextId = "lemur-paas" , path = "/im/user" ) public interface IPaasUserFacade extends IBaseController < PaasUserRequestModel > { }

@FeignClient注解不论是value,name,contextId ,serviceId全都是当做了name,所以在获取nacos地址的地方都是用paasUserFacade去匹配的,根本无法拿到,最终一直跟了N遍代码,也没有找打解决办法,自己还是改了源码FeignClientFactoryBean,在注册bean的地方还是使用value ,并且不用contextId 注册别名,意义不大,还重名,把contextId 当做服务Id,feign是通过target来做地址解析的,所以只要把target的url地址改为lemur-paas/im/user就可以了

<T> T getTarget() { FeignContext context = this.applicationContext.getBean(FeignContext.class); Feign.Builder builder = feign(context); if (!StringUtils.hasText(this.url)) { if (StringUtils.hasText(this.contextId) && !this.name.startsWith("http")){ this.url = "http://" + this.contextId; }else if (StringUtils.hasText(this.contextId)){ this.url = this.contextId; } else if (!this.name.startsWith("http")) { this.url = "http://" + this.name; } else { this.url = this.name; } this.url += cleanPath(); return (T) loadBalance(builder, context, new HardCodedTarget<>(this.type, this.name, this.url)); } if (StringUtils.hasText(this.url) && !this.url.startsWith("http")) { this.url = "http://" + this.url; } String url = this.url + cleanPath(); Client client = getOptional(context, Client.class); if (client != null) { if (client instanceof LoadBalancerFeignClient) { // not load balancing because we have a url, // but ribbon is on the classpath, so unwrap client = ((LoadBalancerFeignClient) client).getDelegate(); } builder.client(client); } Targeter targeter = get(context, Targeter.class); return (T) targeter.target(this, builder, context, new HardCodedTarget<>(this.type, this.name, url)); }

改完源码之后,算是可以互相调用了

2.fastjson 不支持abstract class ,关键是不报错,直接返回null

因为风铃统一返回的都是Response对象,加泛型,结果怎么调用返回的都是null,看看被调用的服务是有收到请求并返回的,这个只能是客户端的问题了,一开始以为是泛型解析的问题,跟踪了整个调用解析链条

ReflectiveFeign . invoke -> SynchronousMethodHandler . invoke -> executeAndDecode -> decode (解析对象)-> ResponseEntityDecoder . decode -> SpringDecoder . decode -> HttpMessageConverterExtractor . extractData (返回泛型)-> FastJsonHttpMessageConverter (真正的解析器). read ( type , clazz , inputMessage )-> parseObject ( is )

然后发现什么,fastjson无法实例化对象,我去你也报个错啊,把abstract去掉,就正常返回了

3.spring gateway 不支持web

spring gateway 因为使用webflux写的,不是web容器所以不能引入web,在引入feign的时候要把web去掉,不然起不来

<dependency> <groupId> org.springframework.cloud </groupId> <artifactId> spring-cloud-starter-openfeign </artifactId> <exclusions> <exclusion> <artifactId> spring-boot-starter-web </artifactId> <groupId> org.springframework.boot </groupId> </exclusion> </exclusions> </dependency>

同时spring gateway 也不支持读取配置文件,同理原因,像j2cache的文件配置方式就读取不到

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

原文链接:https://blog.csdn.net/CoderYin/article/details/90754547

查看更多关于Feign接口方法返回值设置方式的详细内容...

  阅读:21次