当前位置: 首页 > news >正文

Spring Cloud 服务调用详解:Ribbon 负载均衡与 Feign 声明式接口调用

Spring Cloud 服务调用详解:Ribbon 负载均衡与 Feign 声明式接口调用

在微服务架构中,服务与服务之间的调用是核心功能之一。Spring Cloud 提供了多种调用方式,本文将详细介绍:

  • Ribbon 实现客户端负载均衡的原理与实践
  • Feign 实现声明式服务调用的方式及优势

一、Ribbon 实现客户端负载均衡

1.1 什么是 Ribbon?

Ribbon 是 Netflix 提供的一种客户端负载均衡工具,Spring Cloud 对其进行了封装,允许通过服务名代替真实地址实现服务调用,并具备负载均衡策略(如轮询、随机等)。

1.2 配置 RestTemplate 支持 Ribbon

@Configuration
public class ConsumerConfig {@Bean@LoadBalanced  // 开启负载均衡能力public RestTemplate restTemplate() {return new RestTemplate();}
}

1.3 Controller 使用 RestTemplate 远程调用服务

@RestController
public class ConsumerController {@Autowiredprivate RestTemplate restTemplate;@GetMapping("/index")public String index() {return "consumer远程调用provider:" +this.restTemplate.getForObject("http://provider/index", String.class);}
}

此处调用的是名为 provider 的服务,其真实地址由 Ribbon 根据注册中心获取并选择一个实例。


1.4 修改默认的负载均衡策略(轮询 → 随机)

默认策略是轮询(RoundRobinRule),如果想改为随机:

server:port: 8180
provider:ribbon:NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule

1.5 基于 Nacos 实现权重负载均衡

Ribbon 默认不支持权重负载均衡,但我们可以结合 Nacos 的元数据权重机制,自定义一个 NacosWeightedRule 来实现权重分配。

步骤 1:自定义负载均衡策略类
@Slf4j
public class NacosWeightedRule extends AbstractLoadBalancerRule {@Autowiredprivate NacosDiscoveryProperties nacosDiscoveryProperties;@Overridepublic void initWithNiwsConfig(IClientConfig iClientConfig) {//读取配置文件}@Overridepublic Server choose(Object o) {ILoadBalancer loadBalancer = this.getLoadBalancer();BaseLoadBalancer baseLoadBalancer = (BaseLoadBalancer) loadBalancer;//获取要请求的微服务名称String name = baseLoadBalancer.getName();//获取服务发现的相关APINamingService namingService = nacosDiscoveryProperties.namingServiceInstance();try {Instance instance = namingService.selectOneHealthyInstance(name);log.info("选择的实例是port={},instance={}",instance.getPort(),instance);return new NacosServer(instance);} catch (NacosException e) {e.printStackTrace();return null;}}
}
步骤 2:配置 Ribbon 使用自定义规则
provider:ribbon:NFLoadBalancerRuleClassName: com.consumer.config.NacosWeightedRule

只需在服务提供者的 Nacos 元数据中配置或者网页动态配置:

image-20250620154911590

spring:cloud:nacos:discovery:metadata:weight: 3

即可实现服务实例按权重分配请求。


二、Feign 实现声明式接口调用

相比手动拼接 URL 的 RestTemplateFeign 更加优雅,它通过接口加注解的方式实现远程调用,并集成了 Ribbon 支持负载均衡。

2.1 添加依赖

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

2.2 创建 Feign 接口

@FeignClient("provider")
public interface Feign {@GetMapping("/index")String index();
}

Feign 会根据接口方法定义自动生成远程调用逻辑,服务名称来自注册中心。


2.3 启动类开启 Feign 功能

@SpringBootApplication
@EnableFeignClients
public class ConsumerApplication {public static void main(String[] args) {SpringApplication.run(ConsumerApplication.class, args);}@Bean@LoadBalancedpublic RestTemplate restTemplate() {return new RestTemplate();}
}

2.4 控制层直接注入接口调用

@RestController
public class ConsumerController {@Autowiredprivate Feign providerFeign;@GetMapping("/index")public String index() {return "consumer远程调用provider:" + this.providerFeign.index();}
}

Ribbon + RestTemplate 与 Feign 对比

对比项Ribbon + RestTemplateFeign 声明式调用
使用复杂度需要拼接 URL,逻辑较复杂接口 + 注解,简洁直观
代码维护性不利于维护更高的可读性与扩展性
底层实现Ribbon 实现负载均衡Ribbon + 动态代理
适用场景非常规请求或特殊场景调用推荐用于大部分服务之间通信

总结

  • Ribbon 提供基础的客户端负载均衡能力,适用于灵活调用;
  • Feign 提供声明式接口调用方式,推荐作为主流调用方式;
  • 可结合 Nacos 提供的权重元数据,实现更合理的负载均衡策略;
  • 注意:Spring Cloud 2020 版本起,Ribbon 被官方弃用,推荐使用 Spring Cloud LoadBalancer。

附注:推荐使用 Spring Cloud Alibaba Nacos 搭配 Feign,实现高效稳定的服务通信。

相关参考:

  • Spring Cloud OpenFeign 中文文档
  • Spring Cloud ribbon - 《Spring Cloud中文文档》
  • Nacos 官方文档)

如果你觉得本文对你有帮助,欢迎点赞 👍 收藏 ⭐ 留言 💬!

👉 作者博客持续更新:Spring Boot、微服务、前后端分离实践干货

http://www.lqws.cn/news/449695.html

相关文章:

  • 【MATLAB代码】制导方法介绍与例程——追踪法,适用于二维平面,目标是移动的|附完整源代码
  • C++基础算法————并查集
  • 【Dify精讲】第14章:部署架构与DevOps实践【知识卡片】
  • 虹科案例 | 欣旺达如何实现动力电池测试的长期稳定性+自动化?
  • 【NLP入门系列三】NLP文本嵌入(以Embedding和EmbeddingBag为例)
  • 高效获取速卖通商品实时数据:API 接口开发与接入全流程
  • ReAct
  • 【二进制安全作业】250617课上作业4 - start
  • Linux (2)
  • 【stm32】标准库学习——I2C
  • 指标解读——113页企业信息化成熟度评估指标【附全文阅读】
  • 算法第38天|322.零钱兑换\139. 单词拆分
  • C语言:二分搜索函数
  • 【Java学习笔记】线程基础
  • 【RTSP从零实践】2、使用RTP协议封装并传输H264
  • JetBrains IDE v2025.1 升级,AI 智能+语言支持齐飞
  • Maven 之工程化开发核心指南:插件配置、pom 文件与依赖管理
  • GreatSQL加入openKylin社区,共推开源生态建设
  • keep-alive缓存文章列表案例完整代码(Vue3)
  • Symbol.iterator 详解
  • 《被讨厌的勇气》详细概述
  • 【大数据】java API 进行集群间distCP 报错unresolvedAddressException
  • vue3打包后,图片丢失
  • 【unity游戏开发——热更新】什么是Unity热更新
  • 【精选】基于SpringBoot的宠物互助服务小程序平台开发 微信小程序宠物互助系统 宠物互助小程序平台设计与实现 支持救助发布+领养申请+交流互动功能
  • 无人机不再“盲飞”!用Python搞定实时目标识别与跟踪
  • 【Linux驱动开发 ---- 4.2_平台设备(Platform Devices)概述】
  • 1.容器技术与docker环境部署
  • phpstudy无法启动mysql,一启动就关闭,完美解决
  • PLuTo 编译器示例9-12