在这里插入图片描述

01Feign 简介

Feign 是 Spring Cloud Netflix 中的 声明式 HTTP 客户端,它如同一位贴心的信使,帮我们化繁为简,让服务间的调用变得轻松又高效。

Feign 的核心优势在于:。

• 声明式调用:开发者只需定义接口和注解,无需编写繁琐的 HTTP 请求代码,Feign 会自动帮我们处理底层细节。
• 集成 Ribbon:Feign 内置了 Ribbon 负载均衡器,在微服务架构中能智能地分配请求到不同实例,确保系统高可用。
• 集成 Hystrix:与熔断器 Hystrix 深度融合,为服务调用提供容错保障,就像给系统系上了安全带。
• 注解驱动开发:通过 @FeignClient 定义客户端,@RequestMapping 映射请求,搭配 @PathVariable 等注解灵活处理各种参数。
• 高度可定制:支持自定义编码器、解码器、错误处理逻辑,满足个性化开发需求。

02使用 Feign 调用第三方接口

Feign 不仅能在微服务家族内部穿梭调用,还能跨越边界与第三方接口对话。

只需在 @FeignClient 注解中设定 url 属性,Feign 就能精准地找到目标接口。

@FeignClient(name = "third-party-api", url = "https://api.example.com")
public interface ThirdPartyApi {@GetMapping("/data")String fetchData();
}

注解:

  1. @FeignClient 的 name 属性为 Feign 客户端命名,便于识别。
  2. url 属性直接指定第三方接口的基地址,Feign 会基于此构建完整请求路径。
  3. 接口方法通过 @GetMapping 注解定义具体请求路径和方法,清晰直观。

03Feign 调用第三方接口的负载均衡实现

虽然 Feign 与 Ribbon 的天然适配让其在微服务内负载均衡游刃有余,但面对第三方接口时,我们需要另辟蹊径。

借助 Feign 的拦截器功能,我们能巧妙地植入自定义负载均衡逻辑。

@Component
public class FeignLoadBalancerInterceptor implements RequestInterceptor {private final LoadBalancer loadBalancer;public FeignLoadBalancerInterceptor(LoadBalancer loadBalancer) {this.loadBalancer = loadBalancer;}@Overridepublic void apply(RequestTemplate template) {// 获取当前请求的目标服务String serviceName = template.feignTarget().name();// 从负载均衡器获取最佳服务实例地址String selectedUrl = loadBalancer.selectServiceInstance(serviceName);// 反射更新请求模板的 URL(因 Feign 的 URL 字段被 final 修饰)updateRequestUrl(template, selectedUrl);}private void updateRequestUrl(RequestTemplate template, String newUrl) {try {FieldurlField= template.feignTarget().getClass().getDeclaredField("url");urlField.setAccessible(true);urlField.set(template.feignTarget(), newUrl);} catch (Exception e) {throw newRuntimeException("Failed to update request URL", e);}}
}

注解:

  1. FeignLoadBalancerInterceptor 实现 RequestInterceptor 接口,在 Feign 发起请求前拦截。
  2. 通过 LoadBalancer 组件(需自行实现或集成)获取最优服务实例地址。
  3. 利用反射技术动态更新 Feign 客户端的请求 URL,绕过 final 修饰符限制。

04负载均衡算法的多样化实现

为了满足不同业务场景需求,我们精心设计了多种负载均衡算法,并通过配置灵活切换。

轮询算法(RoundRobin):像接力赛跑一样,依次将请求分配给每个服务实例。

@Component("roundRobin") 
public class RoundRobinLoadBalancer implements LoadBalancer {private Atomic Integer position=newAtomicInteger(0);private final List<String> serviceInstances;@Overridepublic String selectServiceInstance(String serviceName) {intidx= Math.abs(position.getAndIncrement() % serviceInstances.size());return serviceInstances.get(idx);}
}

加权轮询算法(WeightedRoundRobin):根据不同实例的承载能力分配权重,权重越高,分配到请求的概率越大。

@Component("weightedRoundRobin")
public class WeightedRoundRobinLoadBalancer implements LoadBalancer {private final Map<String, Integer> weights;private Atomic Integer currentWeight=new AtomicInteger(0);@Overridepublic String selectServiceInstance(String serviceName) {String selectedInstance=null;int maxWeight=0;for (Map.Entry<String, Integer> entry : weights.entrySet()) {currentWeight.addAndGet(entry.getValue());if (currentWeight.get() > maxWeight) {maxWeight = currentWeight.get();selectedInstance = entry.getKey();}}return selectedInstance;}
}

随机算法(RandomAlgo):如同掷骰子般随机挑选服务实例,简单直接。

@Component("randomAlgo")
public class RandomLoadBalancer implements LoadBalancer {private final Randomrandom=newRandom();private final List<String> serviceInstances;@Overridepublic String selectServiceInstance(String serviceName) {return serviceInstances.get(random.nextInt(serviceInstances.size()));}
}

加权随机算法(WeightRandom):在随机挑选的基础上,综合考虑实例权重,权重越大被选中的概率越高。

@Component("weightRandom")
public class WeightedRandomLoadBalancer implements LoadBalancer {private final Map<String, Integer> weights;private final Randomrandom=newRandom();@Overridepublic String selectServiceInstance(String serviceName) {int totalWeight= weights.values().stream().mapToInt(Integer::intValue).sum();int randomWeight= random.nextInt(totalWeight);for (Map.Entry<String, Integer> entry : weights.entrySet()) {randomWeight -= entry.getValue();if (randomWeight <= 0) {return entry.getKey();}}retur nnull;}
}

05定义 Feign 接口实现负载均衡调用

@FeignClient(name = "api-service", url = "http://api.example.com",configuration = FeignLoadBalancerConfig.class)
public interface ApiServiceFeign {@GetMapping("/query")Response fetchData(@RequestParam("param1") String param1,@RequestParam("param2") String param2);
}

注解:

  1. @FeignClient 注解的 configuration 属性引入负载均衡相关配置,激活自定义拦截器。
  2. 接口方法通过 @GetMapping 定义具体请求路径和参数映射,Feign 会自动将方法参数转换为 HTTP 请求参数。

06配置细节与熔断保障

在 application.yml 中合理配置,确保负载均衡与熔断降级协同工作:

spring:cloud:openfeign:circuitbreaker:enabled:true# 启用熔断功能okhttp:enabled:true# 启用 OkHttp 客户端httpclient:enabled:true# 启用 Apache HttpClientmax-connections:300# 最大连接数max-connections-per-route:100# 每路由最大连接数load:
balance:algorithm:com.example.loadbalancer.RoundRobin# 指定负载均衡算法instances:-http://server1.example.com:8080-http://server2.example.com:8081-http://server3.example.com:8082weights:http://server1.example.com:8080:1http://server2.example.com:8081:2http://server3.example.com:8082: 3

通过上述设计与实现,我们成功赋予 Feign 调用第三方接口的负载均衡能力,并通过熔断降级保障系统稳定性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/87432.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/87432.shtml
英文地址,请注明出处:http://en.pswp.cn/web/87432.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

k8s pod调度基础

目录 一&#xff1a;replication controller和replicaset 1&#xff1a;replication controller replication controller的使用示例。 2&#xff1a;标签与标签选择器 &#xff08;1&#xff09;标签 &#xff08;2&#xff09;标签选择器 &#xff08;3&#xff09;标签…

学习者的Python项目灵感

一、实用工具类 - 文件批量重命名工具 用 os 模块实现按规则&#xff08;如添加日期、序号、替换关键词&#xff09;批量重命名文件&#xff0c;适合处理大量图片/文档。 - 简易待办事项管理器&#xff08;To-Do List&#xff09; 用 tkinter 或 PyQt 做GUI界面&#xff0c;…

gRPC服务发现

基于 etcd 实现的服务发现&#xff0c;按照非规范化的 etcd key 实现&#xff0c;详细见代码注释。 package discoveryimport ("context""encoding/json""fmt""go.etcd.io/etcd/api/v3/mvccpb"clientv3 "go.etcd.io/etcd/client/…

基于Linux的Spark本地模式环境搭建实验指南

一、实验目的 掌握Spark本地模式的安装与配置方法验证Spark本地环境是否搭建成功了解Spark基本操作和运行原理 二、实验环境准备 操作系统&#xff1a;Linux&#xff08;推荐ubuntu&#xff09;Java环境&#xff1a;JDK 1.8或以上版本内存&#xff1a;至少4GB&#xff08;推…

数学建模_时间序列

什么是时间序列时间序列预测方法/模型条件&#xff1a;非白噪音平稳平稳性评估不平稳变成平稳然后用ARIMA模型确定p,qAR模型(ARMA特例)MA模型(ARMA特例)ARMA模型(普适)灰色模型神经网络/LSTM组合预测模型向量数据预测结果和为1的情况什么是时间序列 省略具体图形例子 时间序列…

linux用rpm包升级sudo包为sudo-1.9.17-2版本

rpm下载地址&#xff1a; https://www.sudo.ws/dist/packages/1.9.17p1/ 备注&#xff1a;其他压缩包下载地址&#xff1a;https://www.sudo.ws/download.html sudo-1.9.17-2.el7.x86_64.rpm 检查一下&#xff0c;本地sudo版本&#xff0c;执行&#xff1a;sudo -V 或者sudo -…

【开源项目】一款真正可修改视频MD5工具视频质量不损失

文章目录 视频MD5修改工具 🎬📋 目录✨ 功能特点💻 系统要求🏗️ 设计架构🔬 技术原理💻 核心代码1. 视频MD5修改核心逻辑2. 前端异步处理代码3. 错误处理与日志记录📥 安装方法方法一:直接下载方法二:使用本地服务器📚 使用教程基本使用步骤高级使用技巧📁…

Day05: Python 中的并发和并行(1)

理解 Python 中的线程和进程 理解线程和进程是实现在 Python 中并发和并行的基础。这种知识使你能够编写能够看似同时执行多个任务的程序&#xff0c;从而提高性能和响应能力。本课程将深入探讨线程和进程的核心概念、它们的区别&#xff0c;以及它们如何为更高级的并发技术奠…

Spring Boot 集成 MinIO 实现分布式文件存储与管理

Spring Boot 集成 MinIO 实现分布式文件存储与管理 一、MinIO 简介 MinIO 是一个高性能的分布式对象存储服务器&#xff0c;兼容 Amazon S3 API。它具有以下特点&#xff1a; 轻量级且易于部署高性能&#xff08;读写速度可达每秒数GB&#xff09;支持数据加密和访问控制提供…

从小白入门,基于Cursor开发一个前端小程序之Cursor 编程实践与案例分析

Cursor 编程实践与案例分析 Cursor 编程实践与案例分析 1. 什么是 Cursor&#xff1f; Cursor 是一款面向开发者的 AI 编程助手&#xff0c;集成于本地 IDE&#xff0c;支持自然语言与代码的无缝协作。它不仅能自动补全、重构、查找代码&#xff0c;还能理解业务上下文&#…

一、如何用MATLAB画一个三角形 代码

一、如何用MATLAB画一个三角形 代码在MATLAB中绘制三角形可以通过指定三个顶点的坐标并使用 fill 或 patch 函数实现。以下是详细代码示例&#xff1a;方法1&#xff1a;使用 fill 函数&#xff08;简单填充&#xff09;% 定义三角形的三个顶点坐标 (x, y) x [0, 1, 0.5]; % …

Postman自动化测试提取相应body体中的参数

文章目录Postman自动化测试提取相应body体中的参数1. 示例响应 Body 参数2. 提取响应 Body 参数Postman自动化测试提取相应body体中的参数 上一篇的文中介绍了使用postman自动化测试时从响应的header中提取token参数&#xff0c;很多同学私信问如何从响应体body中提取参数。 有…

vue-39(为复杂 Vue 组件编写单元测试)

实际练习:为复杂 Vue 组件编写单元测试 单元测试对于确保复杂 Vue 组件的可靠性和可维护性至关重要。通过隔离和测试代码的各个单元,您可以在开发过程的早期发现并修复错误,从而构建更健壮和可预测的应用程序。本课程重点介绍为复杂 Vue 组件编写单元测试的实用方面,建立在…

c语言中的函数IV

函数的先后关系 直接把函数放在程序上方是可以的 在实际开发中&#xff0c;我们更希望把main函数放在前面 这样子直接把自己定义的函数放在main函数下方&#xff0c;编译会出现warning和error正确的解决方案是&#xff1a;把函数的头放到main函数上方&#xff0c;这样就能正常…

大模型Decoder-Only深入解析

Decoder-Only整体结构 我们以模型Llama-3.1-8B-Instruct为例&#xff0c;打印其结构如下&#xff08;后面会慢慢解析每一部分&#xff0c;莫慌&#xff09;&#xff1a; LlamaForCausalLM((model): LlamaModel((embed_tokens): VocabParallelEmbedding(num_embeddings128256,…

web网页,在线%电商,茶叶,商城,网上商城系统%分析系统demo,于vscode,vue,java,jdk,springboot,mysql数据库

经验心得 这也是帮之前一客户加了几个功能&#xff0c;需要掌握crud&#xff0c;前后端开发&#xff0c;前后端怎么对接&#xff0c;前后端通讯是以那种格式&#xff0c;把这些掌握后咱们就可以进行网站开发了。后端记好一定要分层开发&#xff0c;不要像老早一起所有代码写到一…

MybatisPlus-05.核心功能-条件构造器

一.条件构造器 我们前面使用的MP功能主要是根据id进行操作的&#xff0c;并未涉及到复杂查询。而根据id所进行的增删改查操作在MP中都有直接的封装。但是遇到复杂的查询条件时&#xff0c;如何使用MP进行操作是我们要考虑的问题。因此MP为我们提供了条件构造器。 在BaseMapper…

ES6从入门到精通:常用知识点

变量声明ES6引入了let和const替代var。let用于声明可变的块级作用域变量&#xff0c;const用于声明不可变的常量。块级作用域有效避免了变量提升和污染全局的问题。let name Alice; const PI 3.1415;箭头函数箭头函数简化了函数写法&#xff0c;且自动绑定当前上下文的this值…

51单片机教程(十一)- 单片机定时器

11、单片机定时器 项目目标 通过定时器/计数器实现流水灯控制。知识要点 定时器的结构。TMOD和TCON;定时/计数器工作方式;定时/计数器编程步骤;1、项目分析 前面的流水灯的时间控制通过空循环语句来实现,定时不是很精确。本章通过用定时器来控制流水灯任务可以实现精确的时…

基于opencv的疲劳驾驶监测系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业多年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了多年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…