【SpringBoot系列】微服务远程调用Open Feign深度学习

简介: 【4月更文挑战第9天】微服务远程调度open Feign 框架学习

[toc]


前言

通常一个服务需要调用 Http 端点,Feign 来自 OpenFeign 项目使得以声明式方式调用 http 端点变得更加容易。Spring 通过其 Spring Cloud OpenFeign 集成了 openfeign 集成。

一、引入Feign Client

feign 的实际项目是 OpenFeign https://github.com/OpenFeign/feign ,Spring自带启动器,将以下依赖项添加到项目

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

我们创建一个名为 feign-serviceA 的服务,该服务将调用 feign-serviceB 。feign-serviceB 公开一个 GET 端点 /hello,它返回一个简单的 hello 响应正文。为了使用来自 feign-serviceA 的 feign 客户端调用端点,我们需要执行以下操作 -

  • 创建一个带有@FeignClient注释的界面,如下所示
@FeignClient(url = "<host:port for serviceB>", name = "serviceB")
public interface ServiceBClient{

    @GetMapping("/hello")
    public ResponseEntity<String> sayHello();
}
  • 为主启动类添加@EnableFeignClients 注解

设置好了,我们可以像任何正常的方法调用一样简单地自动连接ServiceBClient并调用方法sayHello。

二、Feign 配置

Feign 自带有一些组件,这些组件用于调用远程端点并编码/解码请求响应。

  • 客户端 - 要进行 HTTP 调用,需要 http 客户端。默认情况下,openfeign 带有默认客户端。我们可以用 ApacheHttpClient、OkHttpClient 或 ApacheHC5FeignClient 覆盖它。这些客户端是委托客户端的包装器。例如,ApacheHttpClient 包装一个 httpcomponents、httpclient。

  • 解码器 - 需要将 feign Response 转换为 feign 方法返回类型的实际类型。默认情况下,spring 提供了一个 OptionalDecoder,它委托给 ResponseEntityDecoder,后者进一步委托给 SpringDecoder,我们可以通过定义一个 Bean 来覆盖它 解码器 。

  • 编码器 - 我们通过向它传递对象来调用feign方法,需要将其转换为 http 请求正文。默认情况下,spring 提供 SpringEncoder。

除了上面的组件外,feign还支持缓存等功能。我们可以创建一个配置类并覆盖上述组件的默认值。如果我们想覆盖单个组件的默认值,@Feign接受配置参数,我们可以使用它来定义默认值的自定义覆盖。

三、重试机制

Feign 已经支持重试机制。但是,默认情况下,它使用 Retry.NEVER_RETRY 。例如,我们可以创建一个自定义重试器,它将重试任何状态代码> 400的请求,以下是我们的 CustomRetryer 的代码:

public class CustomRetryer extends Retryer.Default{
   
    public CustomRetryer(long period, long maxPeriod, int maxAttempts){
   
        super(period, maxPeriod, maxAttempts);
    }

    @Override
    public void continueOrPropagate(RetryableException e){
   
        log.info("Going to retry for ", e);
        super.continueOrPropagate(e);
    }

    @Override
    public Retryer clone(){
   
        return new CustomRetryer(5,SECONDS.toMillis(1), 5);
    }
}

一个重要的事实是 feign Retry 适用于 IOException 或从某些 errorDecoder 引发的 RetryableException ,以下是自定义解码器的样子:

 @Bean
    public ErrorDecoder errorDecoder(){
   
        return (methodKey, response) -> {
   
            byte[] body = {
   };
            try {
   
                if (response.body() != null) {
   
                    body = Util.toByteArray(response.body().asInputStream());
                }
            } catch (IOException ignored) {
    // NOPMD
            }
            FeignException exception = new FeignException.BadRequest(response.reason(), response.request(), body, response.headers());
            if (response.status() >= 400) {
   
                return new RetryableException(
                        response.status(),
                        response.reason(),
                        response.request().httpMethod(),
                        exception,
                        Date.from(Instant.now().plus(15, ChronoUnit.MILLIS)),
                        response.request());
            }
            return exception;
        };
    }

四、弹性

我们在前面介绍了通过重试实现的一种弹性形式,Spring 对 feign 有 CircuitBreaker 支持,它通过单独的 Feign 构建器 FeignCircuitBreaker.Builder 来实现它,Circuitbreaker 的实际实现来自 resilience4j 库。

五、拦截器

有时我们想通过添加一些额外的信息来修改请求,例如,我们可以为每个请求添加一些标头,我们可以通过使用 RequestInterceptor 来实现这一点,在下面添加了填充标头 userid 的拦截器。

@Bean
public RequestInterceptor userIdRequestInterceptor(){
     return (template) -> {
        template.header("userid", "somerandomtext");
    };
}

feign-serviceB 读取此标头并返回为标头,我们请求curl,我们得到的响应如下:

< HTTP/1.1 200
< connection: keep-alive
< date: Sat, 20 Aug 2022 15:27:47 GMT
< keep-alive: timeout=60
< userid: somerandomtext
< Content-Type: text/plain;charset=UTF-8
< Content-Length: 7
<
* transfer closed with 7 bytes remaining to read

我们看到 userid 已添加到响应中。一个非常有用的拦截器应用程序是当 feign 必须发送 oauth2 访问令牌时。Out of the spring 提供了一个 OAuth2FeignRequestInterceptor,它为每个请求添加访问令牌。

六、负载均衡

从 spring boot 2.4.0 开始,feign 与 spring-cloud-loadbalancer 集成,后者可以从各种服务发现提供商获取客户端 url 信息,并使该信息可供 feign 使用。feign 的使用简化了发出 http 请求的各个方面。在典型的生产环境中,我们可能需要重写多个组件,如客户端、解码器、errorDecoder 等。同样在 Spring 生态系统中,feign 与弹性、负载平衡、指标等很好地集成在一起,这使得我们在微服务架构中可以自动选择它。

小结

本节我们学习了Spring Feign Client,我们介绍了Feign的配置,重试机制,弹性以及拦截器,负载均衡等能力。Spring Feign Client在微服务调度过程中起到了重要的作用,通过Spring Feign Client,我们可以实现对其他服务的调度能力,后面我们会对Spring Boot系列继续进行深入学习,欢迎大家持续关注,如果你还喜欢小编的风格,请点赞+关注,给小编一个三连赞,小编一定会特别开心哒。

目录
相关文章
|
7月前
|
Dubbo Java 应用服务中间件
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
|
2月前
|
JSON Java 数据格式
【微服务】SpringCloud之Feign远程调用
本文介绍了使用Feign作为HTTP客户端替代RestTemplate进行远程调用的优势及具体使用方法。Feign通过声明式接口简化了HTTP请求的发送,提高了代码的可读性和维护性。文章详细描述了Feign的搭建步骤,包括引入依赖、添加注解、编写FeignClient接口和调用代码,并提供了自定义配置的示例,如修改日志级别等。
113 1
|
3月前
|
前端开发 API 微服务
SpringCloud微服务之间使用Feign调用不通情况举例
SpringCloud微服务之间使用Feign调用不通情况举例
639 2
|
3月前
|
缓存 Java 应用服务中间件
随着微服务架构的兴起,Spring Boot凭借其快速开发和易部署的特点,成为构建RESTful API的首选框架
【9月更文挑战第6天】随着微服务架构的兴起,Spring Boot凭借其快速开发和易部署的特点,成为构建RESTful API的首选框架。Nginx作为高性能的HTTP反向代理服务器,常用于前端负载均衡,提升应用的可用性和响应速度。本文详细介绍如何通过合理配置实现Spring Boot与Nginx的高效协同工作,包括负载均衡策略、静态资源缓存、数据压缩传输及Spring Boot内部优化(如线程池配置、缓存策略等)。通过这些方法,开发者可以显著提升系统的整体性能,打造高性能、高可用的Web应用。
78 2
|
4月前
|
前端开发 Java API
SpringCloud跨微服务的远程调用,如何发起网络请求,RestTemplate
SpringCloud跨微服务的远程调用,如何发起网络请求,RestTemplate
94 2
|
6月前
|
监控 数据管理 Java
智慧城管源码,基于微服务+java+springboot+vue+uniapp开发的城管综合执法系统源码
智慧城管执法系统利用微服务和Java技术提升城市管理水平,涵盖事件处理、投诉、处罚等功能,包含PC和APP源码。系统支持执法APP,便于领导随时随地审批,具备文书模板、地图定位、法规查询等功能。此外,执法办案系统通过监控视频分析事件,实现案件全程闭环管理,包括组织、案件、信用和执法队伍管理,以及法规库等基础支撑。系统旨在优化流程,提高数据管理和效率。
146 3
智慧城管源码,基于微服务+java+springboot+vue+uniapp开发的城管综合执法系统源码
|
5月前
|
负载均衡 微服务
微服务06----Eureka注册中心,微服务的两大服务,订单服务和用户服务,订单服务需要远程调用我们的用,户服务,消费者,如果环境改变,硬编码问题就会随之产生,为了应对高并发,我们可能会部署成一个集
微服务06----Eureka注册中心,微服务的两大服务,订单服务和用户服务,订单服务需要远程调用我们的用,户服务,消费者,如果环境改变,硬编码问题就会随之产生,为了应对高并发,我们可能会部署成一个集
|
5月前
|
消息中间件 负载均衡 Java
最容易学会的springboot gralde spring cloud 多模块微服务项目
最容易学会的springboot gralde spring cloud 多模块微服务项目
|
6月前
|
缓存 Java 微服务
Springboot微服务整合缓存的时候报循环依赖的错误 两种解决方案
Springboot微服务整合缓存的时候报循环依赖的错误 两种解决方案
85 1

热门文章

最新文章