OpenFeign的集成与优化

简介: OpenFeign的集成与优化

背景

最近在做微服务的集成,为了解决各服务间的rpc调用问题,使用到了openFeign,虽然是简单地在springboot项目中集成openFeign,但是里面其实还是有很多需要注意的点,下面就依次列举出来。

依赖

首先我们的springboot版本是2.7.3,spring cloud版本是2021.0.4,下面我们引入openFeign的依赖:

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
    <version>3.1.4</version>
</dependency>
<!-- openFeign负载均衡由ribbon改为loadbalancer-->
<!-- 不加这个依赖会报错:No Feign Client for loadBalancing defined. Did you forget to include spring-cloud-starter-loadbalancer? -->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-loadbalancer</artifactId>
    <version>3.1.4</version>
</dependency>
复制代码

使用openFeign

  • 依赖引入后,我们需要使用openFeign的功能,首先需要在启动类上添加@EnableFeignClients注解;
  • 其次,我们需要开始编写FeignClient:
// api接口,提供给客户端和服务端共用
public interface WalletApi {
  /**
   * 扣钱
   *
   * @param amount
   * @return
   */
  @PostMapping("/dedutMoney")
  Boolean deductMoney(@RequestParam("amount") long amount);
}
// account代表目标服务名称,需要从注册中心找
// contextId作为注入到ioc容器中bean的名称,要保持唯一性
// path对应服务端的@RequestMapping中的内容
@FeignClient(name = "account", contextId = "walletApi", path = "/wallet")
public interface WalletApiClient extends WalletApi {
}
// 服务端代码
@Slf4j
@RestController
@RequestMapping("/wallet")
public class WalletController implements WalletApi {
  @Override
  public Boolean deductMoney(long amount) {
  }
}
复制代码
  • 上面的代码把客户端和服务端抽象出了公共的一个接口,这样的话,我们就可以避免两边都要写requestMapping这一套,也无形之中让生产者和消费者直接达成了共识,避免很多无效沟通。

至此,openFeign的集成就算是告一段落了,项目中就可以正常使用了。下面我们继续openFeign的优化:

链接池

默认的情况下,openFeign使用的上是HttpURLConnection发起请求,具体代码可查看feign.Client.Default类实现,也就是说,openFeign每次需要创建一个新的请求,而不是使用的链接池,所以我们的需要替换掉这个默认的实现,改用一个有链接池的实现。

  • 添加依赖
    我们打算把HttpURLConnection实现替换成okhttp的实现
<!-- 替换默认的HttpURLConnection,改为okhttp,并添加链接池-->
<dependency>
    <groupId>io.github.openfeign</groupId>
    <artifactId>feign-okhttp</artifactId>
    <version>11.9.1</version>
</dependency>
复制代码
  • 通过javaConfig的方式把okhttp的实现引入进来
import feign.Feign;
import feign.Logger;
import feign.okhttp.OkHttpClient;
import lombok.Getter;
import lombok.Setter;
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.cloud.commons.httpclient.OkHttpClientConnectionPoolFactory;
import org.springframework.cloud.commons.httpclient.OkHttpClientFactory;
import org.springframework.cloud.openfeign.FeignAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import javax.annotation.PreDestroy;
import java.util.concurrent.TimeUnit;
/**
 * @author zouwei
 * @className FeignConfig
 * @date: 2022/9/18 19:12
 * @description:
 */
@Configuration
@ConditionalOnClass(Feign.class)
@AutoConfigureBefore(FeignAutoConfiguration.class)
public class FeignConfig {
  @Getter
  @Setter
  @Configuration
  @ConfigurationProperties(prefix = "feign.okhttp")
  @ConditionalOnProperty(name = "feign.okhttp.enabled", havingValue = "true")
  protected static class OkHttpProperties {
    boolean followRedirects = true;
    // 链接超时时间,单位毫秒
    int connectTimeout = 5000;
    boolean disableSslValidation = false;
    // 读超时,单位毫秒
    int readTimeout = 5000;
    // 写超时,单位毫秒
    int writeTimeout = 5000;
    // 是否自动重连
    boolean retryOnConnectionFailure = true;
    // 最大空闲链接
    int maxIdleConnections = 10;
    // 默认保持5分钟
    long keepAliveDuration = 1000 * 60 * 5L;
  }
  /**
   * 配置okhttp以及对应的链接池
   */
  @Configuration(
      proxyBeanMethods = false
  )
  @ConditionalOnClass({OkHttpClient.class})
  @ConditionalOnMissingBean({okhttp3.OkHttpClient.class})
  @ConditionalOnProperty({"feign.okhttp.enabled"})
  protected static class OkHttpFeignConfiguration {
    private okhttp3.OkHttpClient okHttpClient;
    protected OkHttpFeignConfiguration() {
    }
    @Bean
    public okhttp3.OkHttpClient client(OkHttpClientFactory httpClientFactory, OkHttpProperties properties, OkHttpClientConnectionPoolFactory connectionPoolFactory) {
      this.okHttpClient = httpClientFactory.createBuilder(properties.isDisableSslValidation())
          // 链接超时时间
          .connectTimeout(properties.getConnectTimeout(), TimeUnit.MILLISECONDS)
          // 是否禁用重定向
          .followRedirects(properties.isFollowRedirects())
          //设置读超时
          .readTimeout(properties.getReadTimeout(), TimeUnit.MILLISECONDS)
          //设置写超时
          .writeTimeout(properties.getWriteTimeout(), TimeUnit.MILLISECONDS)
          // 链接失败是否重试
          .retryOnConnectionFailure(properties.isRetryOnConnectionFailure())
          //链接池
          .connectionPool(connectionPoolFactory.create(properties.getMaxIdleConnections(), properties.getKeepAliveDuration(), TimeUnit.MILLISECONDS))
          .build();
      return this.okHttpClient;
    }
    @PreDestroy
    public void destroy() {
      if (this.okHttpClient != null) {
        this.okHttpClient.dispatcher().executorService().shutdown();
        this.okHttpClient.connectionPool().evictAll();
      }
    }
  }
}
复制代码
  • 更改配置
feign:
  # 不使用httpclient,改用okhttp
  httpclient:
    enabled: false
  okhttp:
    enabled: true
    # 是否禁用重定向
    follow-redirects: true
    connect-timeout: 5000
    # 链接失败是否重试
    retry-on-connection-failure: true
    read-timeout: 5000
    write-timeout: 5000
    # 最大空闲数量
    max-idle-connections: 5
    # 生存时间
    keep-alive-duration: 15000
复制代码

这样我们就把openFeign的请求发送改造成链接池了,避免了每次请求都创建HttpURLConnection对象;

开启请求压缩功能

为了更好地减少请求发送的时间,我们可以针对请求数据进行压缩处理,openFeign也内置了压缩功能,不过需要我们自己开启:

feign:
  # 开启压缩功能
  compression:
    request:
      enabled: true
      mime-types: text/xml,application/xml,application/json
      min-request-size: 2048
    response:
      enabled: true
复制代码

配置超时时间

我们还可以给指定的FeignClient指定对应的超时时间,因为并不是所有的服务超时时间都是统一的,有些特殊的业务场景需要针对性地设置超时时间:

feign:
  client:
    config:
      # 设置超时,囊括了okhttp的超时,okhttp属于真正执行的超时,openFeign属于服务间的超时
      # 设置全局超时时间
      default:
        connectTimeout: 2000
        readTimeout: 5000
      # 针对特定contextId设置超时时间
      walletApi:
        connectTimeout: 1000
        readTimeout: 2000
复制代码

添加LoadBalancerCacheManager

在项目启动过程中,会出现警告:LoadBalancerCacheManager not available, returning delegate without caching.,说明LoadBalancerCacheManager没有开启,我们需要加入以下依赖并开启缓存:

<!-- 解决项目启动警告:LoadBalancerCacheManager not available, returning delegate without caching.-->
<!-- 如果注册中心有自己的缓存,那么就可以禁用loadbalancer的缓存-->
<dependency>
    <groupId>com.github.ben-manes.caffeine</groupId>
    <artifactId>caffeine</artifactId>
    <version>3.1.1</version>
</dependency>
<!-- 解决项目启动警告:Spring Cloud LoadBalancer is currently working with the default cache. You can switch to using Caffeine cache, by adding it and org.springframework.cache.caffeine.CaffeineCacheManager to the classpath.-->
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-context-support</artifactId>
</dependency>
复制代码

另外,我们还需要通过配置开启缓存:

spring:
  cloud:
    loadbalancer:
      cache:
        # 开启缓存,如果注册中心有自己的缓存,那么就可以禁用loadbalancer的缓存
        enabled: true
        # 过期时间10s
        ttl: 10
        # 容量256M
        capacity: 256
        caffeine:
          #          initialCapacity=[integer]: sets Caffeine.initialCapacity.
          #          maximumSize=[long]: sets Caffeine.maximumSize.
          #          maximumWeight=[long]: sets Caffeine.maximumWeight.
          #          expireAfterAccess=[duration]: sets Caffeine.expireAfterAccess(long, java.util.concurrent.TimeUnit).
          #          expireAfterWrite=[duration]: sets Caffeine.expireAfterWrite(long, java.util.concurrent.TimeUnit).
          #          refreshAfterWrite=[duration]: sets Caffeine.refreshAfterWrite(long, java.util.concurrent.TimeUnit).
          #          weakKeys: sets Caffeine.weakKeys().
          #          weakValues: sets Caffeine.weakValues().
          #          softValues: sets Caffeine.softValues().
          #          recordStats: sets Caffeine.recordStats().
          #         initialCapacity初始化键值对的数量
          spec: initialCapacity=500,expireAfterWrite=5s
复制代码

至此,我们基本上已经把OpenFeign的几个优化点列举完毕,感兴趣的小伙伴也可以自己尝试一下。


相关实践学习
部署高可用架构
本场景主要介绍如何使用云服务器ECS、负载均衡SLB、云数据库RDS和数据传输服务产品来部署多可用区高可用架构。
负载均衡入门与产品使用指南
负载均衡(Server Load Balancer)是对多台云服务器进行流量分发的负载均衡服务,可以通过流量分发扩展应用系统对外的服务能力,通过消除单点故障提升应用系统的可用性。 本课程主要介绍负载均衡的相关技术以及阿里云负载均衡产品的使用方法。
相关文章
|
18天前
|
存储 测试技术 持续交付
自动化测试与持续集成/持续交付(CI/CD):优化软件开发流程的利器
自动化测试与持续集成/持续交付(CI/CD)是现代软件开发中至关重要的环节,通过将自动化测试与持续集成/持续交付相结合,可以实现开发流程的高效优化,提高软件质量和交付速度。本文将探讨自动化测试与CI/CD的概念、原理及其在软件开发中的重要性,以及如何实施这些技术以提升团队的协作效率和软件交付质量。
72 1
|
4天前
|
敏捷开发 存储 测试技术
探索自动化测试在持续集成中的应用与优化策略
【5月更文挑战第29天】 随着敏捷开发和持续集成(CI)的普及,自动化测试已成为确保软件质量和加快交付速度的关键。本文将深入探讨自动化测试在持续集成环境中的实际应用,并提出一系列优化策略。我们将分析自动化测试流程中常见的挑战,如测试用例管理、测试数据准备、并行执行以及结果反馈机制等,并基于这些挑战提出相应的解决方案。文章旨在帮助读者理解如何通过优化自动化测试流程来提升软件开发的效率和质量。
|
5天前
|
敏捷开发 监控 测试技术
深入探究持续集成在软件测试中的应用与优化
【5月更文挑战第28天】随着敏捷开发和DevOps文化的普及,持续集成(CI)已成为软件开发周期不可或缺的一部分。它不仅加速了开发过程,而且提高了代码质量和项目透明度。然而,在实际操作中,如何有效实施CI以提升软件测试效率并非易事。本文将详细探讨持续集成在软件测试领域的应用,分析其面临的挑战,并提出一系列优化策略,以期帮助测试人员和开发团队更好地利用CI实现质量保证的目标。
|
6天前
|
安全 数据管理 测试技术
网络安全与信息安全:防范漏洞、加强加密与提升安全意识深入探索自动化测试框架的设计原则与实践应用化测试解决方案。文章不仅涵盖了框架选择的标准,还详细阐述了如何根据项目需求定制测试流程,以及如何利用持续集成工具实现测试的自动触发和结果反馈。最后,文中还将讨论测试数据管理、测试用例优化及团队协作等关键问题,为读者提供全面的自动化测试框架设计与实施指南。
【5月更文挑战第27天】 在数字化时代,网络安全与信息安全已成为维护国家安全、企业利益和个人隐私的重要环节。本文旨在分享关于网络安全漏洞的识别与防范、加密技术的应用以及提升安全意识的重要性。通过对这些方面的深入探讨,我们希望能为读者提供一些实用的建议和策略,以应对日益严峻的网络安全挑战。 【5月更文挑战第27天】 在软件开发周期中,自动化测试作为保障软件质量的关键步骤,其重要性日益凸显。本文旨在剖析自动化测试框架设计的核心原则,并结合具体案例探讨其在实际应用中的执行策略。通过对比分析不同测试框架的优缺点,我们提出一套高效、可扩展且易于维护的自动
|
18天前
|
缓存 关系型数据库 数据库
【Docker 专栏】Docker 与容器化数据库的集成与优化
【5月更文挑战第9天】本文探讨了Docker与容器化数据库集成的优势,如快速部署、环境一致性、资源隔离和可扩展性,并列举了常见容器化数据库(如MySQL、PostgreSQL和MongoDB)。讨论了集成方法、注意事项、优化策略,包括资源调整、缓存优化和监控告警。此外,强调了数据备份、恢复测试及性能评估的重要性。未来,随着技术发展,二者的集成将更紧密,为数据管理带来更多可能性。掌握此技术将应对数字化时代的机遇与挑战。
【Docker 专栏】Docker 与容器化数据库的集成与优化
|
18天前
|
敏捷开发 数据管理 测试技术
探索自动化测试在持续集成环境中的优化策略
【5月更文挑战第6天】 本文旨在深入剖析自动化测试在持续集成(CI)环境中所面临的挑战,并提出一系列创新的优化策略。通过对现代软件开发过程中自动化测试角色的分析,我们揭示了在快速迭代和部署的背景下,如何通过改进测试框架、选择合适的测试工具、以及实施数据驱动测试等手段来提高测试效率和准确性。文章不仅聚焦于技术层面的解决方案,还探讨了团队协作和流程管理对提升自动化测试效能的重要性。
|
18天前
|
传感器 数据采集 数据处理
MATLAB热传导方程模型最小二乘法模型、线性规划对集成电路板炉温优化
MATLAB热传导方程模型最小二乘法模型、线性规划对集成电路板炉温优化
|
18天前
|
敏捷开发 测试技术 持续交付
深入探究持续集成在软件测试中的应用与优化
【4月更文挑战第28天】随着敏捷开发模式的普及,持续集成(Continuous Integration, CI)已成为软件开发工作流中不可或缺的一环。本文将深入探讨CI在软件测试领域的关键作用,分析其如何提升测试效率和质量,并指出实践中常见的挑战及解决策略。通过对自动化测试流程、测试驱动开发(TDD)以及持续部署(CD)等关键技术的综合运用,揭示了构建高效、可靠软件系统的方法论。
|
18天前
|
敏捷开发 Devops 测试技术
深入探究持续集成在软件测试中的应用与优化
【4月更文挑战第28天】随着敏捷开发和DevOps文化的兴起,持续集成(CI)已经成为现代软件开发不可或缺的一环。本文将探讨持续集成在软件测试领域的关键作用,分析其如何提高测试效率、确保产品质量,并指出实施过程中可能遇到的挑战及相应的解决策略。通过案例研究和最佳实践的分享,旨在为读者提供一套系统的持续集成优化方案,以支持更高效、更可靠的软件发布流程。
|
18天前
|
机器学习/深度学习 存储 算法
PYTHON集成机器学习:用ADABOOST、决策树、逻辑回归集成模型分类和回归和网格搜索超参数优化
PYTHON集成机器学习:用ADABOOST、决策树、逻辑回归集成模型分类和回归和网格搜索超参数优化