「Feign」OpenFeign入门以及远程调用

一、OpenFeign介绍

OpenFeign是 种声明式,模版化的HTTP客户端。使 OpenFeign进 远程调 时,开发者完全感知不到这是在进 远程调 , 是像在调 本地 法 样。使 式是注解+接 形式,把需要调 的远程接 封装到接 当中,映射地址为远程接 的地址。在启动SpringCloud应 时,Feign会扫描标有@FeignClient注解的接 , 成代理并且注册到Spring容器当中。 成代理时Feign会为每个接 法创建 个RequestTemplate对象,该对象封装HTTP请求需要的全部信息,请求参数名、请求 法等信息都是在这个过程中确定的,模版化就体现在这 。

二、OpenFeign的使用

  • 搭建前置环境,在pom.xml文件中引入依赖,可以选择使用注册中心或者配置中心

org.springframework.cloud spring-cloud-dependencies 2020.0.3 pom import org.springframework.cloud spring-cloud-starter-consul-config org.springframework.cloud spring-cloud-starter-consul-discovery org.springframework.boot spring-boot-starter-actuator org.springframework.cloud spring-cloud-starter-openfeign

1.使用注册中心

  • 使 注册中 ,将服务注册到consul(nacos),调 者拿到被调 服务的地址端 进 调

spring.cloud.consul.host=192.168.0.124#consul地址spring.cloud.consul.port=8080#端 号spring.cloud.consul.discovery.service-name=service-test-01#服务名称spring.cloud.consul.discovery.health-check-interval=1m#健康检查间隔时间server.port=10000#服务端 号

  • 在配置类上开启服务发现以及允许远程调

@EnableDiscoveryClient //开启服务发现@EnableFeignClients //开启服务调 ,只需要在调 开启即可

  • 服务运 之后可以在consul的UI界 看到运 的服务,consul会定时检查服务的健康状态
  • 创建远程调用接口

@FeignClient(“serviceName”)public interface Service2Remote { /** 这 有 定义解码器对远程调 的结果进 解析,拿到真正的返回类型,所以接 返回值类型和远程接 返回类型保持 致 **/ @PostMapping(“/page”) List pageQuestion(PageQuestionReq req);}

  • 简单使用

@RestController@RequestMapping(“/service/remote”)public class RemoteController { @Autowired private Service2Remote service2Remote; @PostMapping(“/getQuestionList”) public List getQuestionList(@RequestBody PageQuestionReq req){ List result = service2Remote.pageQuestion(req); //对拿到的数据进 处理… return result; }}

2.使用配置中心

  • 将请求的URL写在配置中 进 读取修改配置 件

spring.cloud.consul.config.format=KEY_VALUE#consul 持yaml格式和Key-value形式spring.cloud.consul.config.enabled=true#开启配置spring.cloud.consul.config.prefixes=glab/plat/wt/application/test#consul配置存放的外层 件夹 录spring.cloud.consul.config.default-context=config# 级 件夹spring.cloud.consul.config.watch.delay=1000#轮询时间spring.cloud.consul.discovery.enabled=false#关闭注册remote.url=www.baidu.com#请求地址

  • 创建远程调用接口

@FeignClient(name = “service2RemoteByUrl”,url = “${remote.url}”) //name需要配置,URL从配置中 读取public interface Service2RemoteByUrl { @PostMapping(“/page”) List pageQuestion(PageQuestionReq req);}

3.自定义解码器(编码器

// 定义解码器实现Decoder接 ,重写decode 法即可,根据具体需求进 编写//如果是 定义编码器,需要实现Encoder接 ,重写encode 法public class FeignDecoder implements Decoder { @Override public Object decode(Response response, Type type) throws IOException,DecodeException, FeignException { if (response.body() == null){ throw new DecodeException(ErrorEnum.EXECUTE_ERR.getErrno(),”没有获取到有效结果值”,response.request()); } // 拿到值 String result = Util.toString(response.body().asReader(Util.UTF_8)); Map resMap = null; try { resMap = JSON.parseObject(result, Map.class); } catch (Exception e) { //返回结果是字符串 return result; }}

4.远程调用携带Cookie

  • 由于feign调 是新创建 个Request,因此在请求时不会携带 些原本就有的信息,例如Cookie,因此需要 定义RequestInterceptor对Request进 额外设置, 般情况下,写 Cookie是 较常 的做法,如下设置

@Configurationpublic class BeanConfig { @Bean public RequestInterceptor requestInterceptor(){ return template -> { ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequest request = attributes.getRequest(); //此处可以根据业务 具体定制携带规则 String data = request.getParameter(“data”); String code = null; try { //这 需要转码,否则会报错 code = URLEncoder.encode(data, “UTF-8”); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } template.query(“data”,code); //请求头中携带Cookie String cookie = request.getHeader(“Cookie”); template.header(“Cookie”,cookie); }; } @Bean public Decoder decoder(){ return new FeignDecoder(); }}

三、调用流程解析

//在使 EnableFeignClients开启feign功能时,点击进 会看到该注解是通过ImportFeignClientsRegistrar类 效的,其中有个 法//registerBeanDefinitions执 两条语句registerDefaultConfiguration(metadata, registry); //加载默认配置信息registerFeignClients(metadata, registry); //注册扫描标有FeignClient的接 //关注registerFeignClients 法for (String basePackage : basePackages) { candidateComponents.addAll(scanner.findCandidateComponents(basePackage)); //在basePackage路径下扫描并添加标有FeignClient的接 }for (BeanDefinition candidateComponent : candidateComponents) { //遍历 if (candidateComponent instanceof AnnotatedBeanDefinition) { registerClientConfiguration(registry, name, attributes.get(“configuration”)); // registerFeignClient(registry, annotationMetadata, attributes); //注册到Spring容器当中, 法详细在FeignClientsRegistrar类当中 }}//在对feign调 时进 断点调试//在 成Feign远程接 的代理类时,调 处理器是Feign提供的FeignInvocationHandlerpublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if (“equals”.equals(method.getName())) { //equals,hashCode,toString三个 法直接本地执 } else if (“hashCode”.equals(method.getName())) { return hashCode(); } else if (“toString”.equals(method.getName())) { return toString(); } //执 法对应的 法处理器MethodHandler,这个接 是Feign提供的,与InvocationHandler 任何关系,只有 个invoke 法 return dispatch.get(method).invoke(args);}//点进上 的invoke 法public Object invoke(Object[] argv) throws Throwable { //创建 个request模版 RequestTemplate template = buildTemplateFromArgs.create(argv); while (true) { try { return executeAndDecode(template, options); //创建request执 并且解码 } }}Object executeAndDecode(RequestTemplate template, Options options) throws Throwable { Request request = targetRequest(template); //创建Request并增强 Response response = client.execute(request, options); //执 调用请求,不再继续分析了 response = response.toBuilder().request(request).requestTemplate(template).build(); //如果有重写解码器,使 定义的解码器,feign默认使 SpringEncoder if (decoder != null) return decoder.decode(response, metadata.returnType()); } Request targetRequest(RequestTemplate template) { //如果 定义了RequestInterceptor,在这 可以对Request进 增强 for (RequestInterceptor interceptor : requestInterceptors) { //执 定义的apply 法 interceptor.apply(template); } //创建Request return target.apply(template);}

四、补充

  • 关于Client接 的实现类,使 注册中 和使 配置中 其流程稍有区别

//使 配置中 拿url 式进 调 ,使 的是Client的默认内部实现类 Default ,其中Default使 的是HttpURLConnection进 Http请求的HttpURLConnection connection = convertAndSend(request, options);//如果使 的是服务发现,使 的使 Client的实现类FeignBlockingLoadBalancerClient,它会去根据配置的服务名去注册中 查找服务的IP地址和端 号,执 使 的仍然是默认实现类Default,通过HttpURLConnection请求//FeignBlockingLoadBalancerClient,根据服务名称查找服务IP地址、端 88 ServiceInstance instance = loadBalancerClient.choose(serviceId, lbRequest);//具体实现 法,BlockingLoadBalancerClient类中 145 Response loadBalancerResponse = Mono.from(loadBalancer.choose(request)).block();//还有其他实现Client接 的客户端,例如ApacheHttpClient,ApacheHttpClient带有连接池功能,具有优秀的HTTP连接复 能 ,需要通过引 依赖来使

郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
(0)
用户投稿
上一篇 2022年6月13日
下一篇 2022年6月13日

相关推荐

  • MYSQL存储过程即常用逻辑知识点总结

    Mysql存储过程 1.创建存储过程语法(格式) DELIMITER $CREATE PROCEDURE 存储过程名A(IN 传入参数名a INT,IN 传入参数名b VARCHA…

    2022年6月18日
  • 国内首个政务网络安全运营中心亮相

    托管服务不是新事物,各行各业都有,有服务器托管、主机托管。云服务实际上是IaaS、 PaaS、SaaS三个不同层次的服务,即基础设施托管服务、平台托管服务、软件托管服务。今天我们要…

    2022年6月15日
  • 微信实名认证怎么更改(更改微信实名认证教程)

    微信是我们生活中必不可少的社交软件,我们可以利用它进行付款以及聊天等。在微信中我们还可以对自己的身份进行实名认证,从而增加账号的真实性,下面就一起来看看认证的方法吧! 微信身份实名…

    2022年5月1日
  • 浅析Web3.0

    首先了解什么是web3.0? web3.0这个词是以太坊联合创始人Gavin Wood在 2014 年提出的,它是基于区块链的去中心化在线生态系统。直白点说就是互联网的下一个形态,…

    2022年6月14日
  • 美国近30天窃取970亿条全球数据!严防数据霸权

    一份6月13日由 “安在”安全领域新媒体发布的网络安全报告显示,美国军方和政府相关部门在最近30天内,通过互联网窃取了超过 970 亿条全球互联网数据和 1240 亿条电话记录,这…

    2022年6月15日
  • 什么是“全息”

    全息(hologram)是一个合成词,源自希腊语:holos(全部)+gramma(信息),顾名思义,它是记录并重现物体全部光学信息的一种技术。 那么,光学信息包括什么呢? 我们都…

    2022年8月19日
  • 北京城市副中心运河骑游季启动,将组织多场特色线路骑行

    骑游季活动吸引了近千名骑行爱好者参与。 图/通州区体育局 新京报讯 (首席记者孙海光)7月24日,“喜迎二十大”北京城市副中心运河骑游季活动在城市绿心森林公园拉开帷幕。来自通州区妇…

    2022年8月8日
  • 内容审核标准评测助力突破产业难题

    央广网北京6月22日消息(记者卜叶)近年来,线上办公、线上娱乐被越来越多的人接受,与此同时,也为互联网内容的监管和治理带来很大的挑战。央广网邀请中国信通院云大所内容科技部工程师陈文…

    2022年6月23日
  • 补齐短板夯实基础-开源IM项目OpenIM关于初始化登录好友接口介绍

    OpenIM文档方面的建设一直远远落后于开发, 也经常被开发者诟病,在接下来一周的时间里,我们重点补齐文档,让开发者更轻松接入。由于app sdk底层都是go来实现,所以本文先写一…

    2022年6月27日
  • 一个人有没有福气,可以从他的鞋子看出来

    有人说:“你穿什么样的鞋,就决定了你要走什么样的路。” 每个人都需要穿鞋,夏天穿凉鞋,冬天穿棉鞋,即使在家不出门也要穿拖鞋。小小的一双鞋,不仅保护了人的脚,还承载了行走的重任。 俗…

    2022年8月16日

联系我们

联系邮箱:admin#wlmqw.com
工作时间:周一至周五,10:30-18:30,节假日休息