OpenFeign的集成与优化

本文涉及的产品
应用型负载均衡 ALB,每月750个小时 15LCU
网络型负载均衡 NLB,每月750个小时 15LCU
简介: 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的几个优化点列举完毕,感兴趣的小伙伴也可以自己尝试一下。


相关实践学习
SLB负载均衡实践
本场景通过使用阿里云负载均衡 SLB 以及对负载均衡 SLB 后端服务器 ECS 的权重进行修改,快速解决服务器响应速度慢的问题
负载均衡入门与产品使用指南
负载均衡(Server Load Balancer)是对多台云服务器进行流量分发的负载均衡服务,可以通过流量分发扩展应用系统对外的服务能力,通过消除单点故障提升应用系统的可用性。 本课程主要介绍负载均衡的相关技术以及阿里云负载均衡产品的使用方法。
相关文章
|
4月前
|
数据采集 XML JavaScript
如何优化 Selenium 和 BeautifulSoup 的集成以提高数据抓取的效率?
如何优化 Selenium 和 BeautifulSoup 的集成以提高数据抓取的效率?
|
1月前
|
存储 Java 开发工具
【三方服务集成】最新版 | 阿里云OSS对象存储服务使用教程(包含OSS工具类优化、自定义阿里云OSS服务starter)
阿里云OSS(Object Storage Service)是一种安全、可靠且成本低廉的云存储服务,支持海量数据存储。用户可通过网络轻松存储和访问各类文件,如文本、图片、音频和视频等。使用OSS后,项目中的文件上传业务无需在服务器本地磁盘存储文件,而是直接上传至OSS,由其管理和保障数据安全。此外,介绍了OSS服务的开通流程、Bucket创建、AccessKey配置及环境变量设置,并提供了Java SDK示例代码,帮助用户快速上手。最后,展示了如何通过自定义starter简化工具类集成,实现便捷的文件上传功能。
【三方服务集成】最新版 | 阿里云OSS对象存储服务使用教程(包含OSS工具类优化、自定义阿里云OSS服务starter)
|
3月前
|
缓存 前端开发 JavaScript
微前端集成优化:让所有子应用体积更小,加载更快!
【8月更文挑战第17天】微前端集成优化:让所有子应用体积更小,加载更快!
70 1
微前端集成优化:让所有子应用体积更小,加载更快!
|
3月前
|
存储 设计模式 缓存
OpenFeign集成Ribbon负载均衡-过滤和选择服务核心实现
该文章主要介绍了如何在OpenFeign中集成Ribbon以实现负载均衡,并详细分析了Ribbon中服务选择和服务过滤的核心实现过程。文章还涉及了Ribbon中负载均衡器(ILoadBalancer)和负载均衡策略(IRule)的初始化方式。
OpenFeign集成Ribbon负载均衡-过滤和选择服务核心实现
|
3月前
|
缓存 负载均衡 Java
OpenFeign最核心组件LoadBalancerFeignClient详解(集成Ribbon负载均衡能力)
文章标题为“OpenFeign的Ribbon负载均衡详解”,是继OpenFeign十大可扩展组件讨论之后,深入探讨了Ribbon如何为OpenFeign提供负载均衡能力的详解。
OpenFeign最核心组件LoadBalancerFeignClient详解(集成Ribbon负载均衡能力)
|
3月前
|
机器学习/深度学习 存储 前端开发
实战揭秘:如何借助TensorFlow.js的强大力量,轻松将高效能的机器学习模型无缝集成到Web浏览器中,从而打造智能化的前端应用并优化用户体验
【8月更文挑战第31天】将机器学习模型集成到Web应用中,可让用户在浏览器内体验智能化功能。TensorFlow.js作为在客户端浏览器中运行的库,提供了强大支持。本文通过问答形式详细介绍如何使用TensorFlow.js将机器学习模型带入Web浏览器,并通过具体示例代码展示最佳实践。首先,需在HTML文件中引入TensorFlow.js库;接着,可通过加载预训练模型如MobileNet实现图像分类;然后,编写代码处理图像识别并显示结果;此外,还介绍了如何训练自定义模型及优化模型性能的方法,包括模型量化、剪枝和压缩等。
53 1
|
3月前
|
搜索推荐
实现CRM与ERP系统无缝集成,优化客户关系管理
在当今竞争激烈的市场环境中,企业要想保持领先地位,必须高效地管理客户关系并优化内部资源。CRM(客户关系管理)系统与ERP(企业资源规划)系统的无缝集成,为企业提供了一种强大的工具,以实现这一目标
72 2
|
3月前
|
JSON 数据管理 关系型数据库
【Dataphin V3.9】颠覆你的数据管理体验!API数据源接入与集成优化,如何让企业轻松驾驭海量异构数据,实现数据价值最大化?全面解析、实战案例、专业指导,带你解锁数据整合新技能!
【8月更文挑战第15天】随着大数据技术的发展,企业对数据处理的需求不断增长。Dataphin V3.9 版本提供更灵活的数据源接入和高效 API 集成能力,支持 MySQL、Oracle、Hive 等多种数据源,增强 RESTful 和 SOAP API 支持,简化外部数据服务集成。例如,可轻松从 RESTful API 获取销售数据并存储分析。此外,Dataphin V3.9 还提供数据同步工具和丰富的数据治理功能,确保数据质量和一致性,助力企业最大化数据价值。
183 1
|
3月前
|
JavaScript jenkins 持续交付
自动化部署与持续集成:使用Jenkins和Docker优化开发流程
【8月更文挑战第31天】在软件开发的世界里,时间就是一切。本文将引导你通过Jenkins和Docker的强大组合,实现自动化部署和持续集成,让你的开发流程如丝般顺滑。我们将从基础设置开始,逐步深入到构建管道,最终实现一键部署的梦想。准备好让你的开发效率飞跃,一起探索这个令人兴奋的旅程吧!
|
3月前
|
API UED 开发者
如何在Uno Platform中轻松实现流畅动画效果——从基础到优化,全方位打造用户友好的动态交互体验!
【8月更文挑战第31天】在开发跨平台应用时,确保用户界面流畅且具吸引力至关重要。Uno Platform 作为多端统一的开发框架,不仅支持跨系统应用开发,还能通过优化实现流畅动画,增强用户体验。本文探讨了Uno Platform中实现流畅动画的多个方面,包括动画基础、性能优化、实践技巧及问题排查,帮助开发者掌握具体优化策略,提升应用质量与用户满意度。通过合理利用故事板、减少布局复杂性、使用硬件加速等技术,结合异步方法与预设缓存技巧,开发者能够创建美观且流畅的动画效果。
83 0
下一篇
无影云桌面