OpenFeign原理

文章目录

概要

Spring Cloud OpenFeign 是 Spring Cloud 中用于实现声明式 HTTP 客户端的组件。通过它,可以使用类似调用本地方法的方式调用远程 HTTP 服务,极大地简化了微服务架构中服务之间的通信。

原理概述

OpenFeign 的原理主要涉及通过代理类将服务名称替换为具体的 IP:端口 来实现远程服务调用。

  • Feign 生成代理类 :使用 JDK 动态代理为 @FeignClient 接口生成代理,拦截接口方法调用,构建 HTTP 请求。
  • 服务名称替换为 IP:端口 :Feign 会将服务名称(如 serviceA)通过 Ribbon 进行解析,将其替换为对应的 IP:端口
  • 负载均衡:Ribbon 在服务发现的基础上,对多个服务实例进行负载均衡,选择一个合适的服务实例。
  • 发送 HTTP 请求:最终由 Feign 发起 HTTP 请求,调用远程服务并返回结果。

OpenFeign 的基本使用

java 复制代码
@FeignClient(name = "serviceA") // 指定服务名称
public interface ServiceAClient {
    @GetMapping("/api/data")
    String getData();
}

通过注解 @FeignClient,我们定义了一个远程调用接口 ServiceAClient,并通过 getData() 方法发起 GET 请求。Feign 会根据这个接口生成动态代理,并执行远程调用。

核心注解和功能

  • @FeignClient:这个注解用于定义一个 Feign 客户端接口,主要是用来标识接口为 Feign 客户端。
  • @EnableFeignClients :这个注解用于启动 Feign 客户端功能,通常在 Spring Boot 启动类上添加。
java 复制代码
@EnableFeignClients
@SpringBootApplication
public class FeignExampleApplication {
    public static void main(String[] args) {
        SpringApplication.run(FeignExampleApplication.class, args);
    }
}

源码解析流程

@FeignClient 的解析

@FeignClient 注解是 Feign 的核心,它定义了远程调用的接口。FeignClient 在 Spring Boot 启动时会被扫描,并注册为 Spring 容器中的 Bean。具体流程如下:

  1. FeignClientsRegistrar
  • @FeignClient 注解的解析工作由 FeignClientsRegistrar 类完成。FeignClientsRegistrar 实现了 ImportBeanDefinitionRegistrar 接口(
    ImportBeanDefinitionRegistrar 用于在运行时动态地向 Spring 容器中注册 Bean 定义。)

  • FeignClientsRegistrar 类中的 registerBeanDefinitions() 方法中,它扫描带有 @FeignClient 的接口,解析并注册这些接口对应的 Feign 客户端 Bean。

java 复制代码
	@Override
	public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
		registerDefaultConfiguration(metadata, registry);
		registerFeignClients(metadata, registry); // 细看源码
	}
  1. FeignClientFactoryBean

    • 每一个 @FeignClient 注解都会生成一个 FeignClientFactoryBean,该类实现了 FactoryBean 接口,用于动态创建 Feign 客户端。
    • FeignClientFactoryBeangetObject() 方法中会返回一个动态代理类,这个代理类负责将接口调用转化为 HTTP 请求。
    java 复制代码
     @Override
     public Object getObject() {
     	return getTarget(); // 可自己看看源码
     }
Feign 动态代理的实现

Feign 本质上是一个基于 JDK 动态代理的 HTTP 客户端。每次调用接口方法时,都会通过动态代理拦截调用,并将其转换为 HTTP 请求。

  1. 代理的创建

    ReflectiveFeign 继承 Feign 类,ReflectiveFeign 使用 JDK 的 Proxy.newProxyInstance() 方法为标注了 @FeignClient 的接口生成一个代理对象。

  2. FeignInvocationHandler

    每次调用接口时,代理对象会调用 FeignInvocationHandlerinvoke() 方法。该方法中,Feign 会根据接口的注解(如 @GetMapping 等)构建 HTTP 请求,并通过 Client 发送请求。

  3. MethodHandler
    MethodHandler 是一个接口,具体的实现类 SynchronousMethodHandler 负责将方法调用转化为 HTTP 请求,并处理返回结果。

请求的执行与负载均衡
  1. 请求的执行

    Feign 的 Client 接口负责发送 HTTP 请求,默认情况下使用 HttpURLConnection 作为底层 HTTP 客户端。也可以集成 OkHttpApache HttpClient 等其他 HTTP 客户端。

  2. 负载均衡

    在 Spring Cloud 中,Feign 可以与 Ribbon(客户端负载均衡器)集成。当 Feign 调用服务时,Ribbon 会根据配置自动选择一个服务实例,从而实现客户端的负载均衡。

    Feign 的 LoadBalancerFeignClient 包装了原始的 Feign Client,负责从 Ribbon 中获取服务实例并执行请求:

结果的解码(Decoder)

Feign 的 Decoder 接口负责将 HTTP 响应转换为方法的返回值。默认的 Decoder 使用 JacksonGson 来将 JSON 响应解析为对象。你可以通过配置自定义的 Decoder 来改变这个行为。

总结

Spring Cloud OpenFeign 通过一系列注解和动态代理机制将接口调用转化为 HTTP 请求,简化了微服务之间的通信。其核心流程可以总结为以下几个步骤:

  1. @FeignClient 注解标注的接口会在应用启动时被扫描并注册为 Spring Bean。
  2. FeignClientFactoryBean 创建 Feign 客户端的动态代理对象。
  3. Feign 的动态代理将接口方法调用拦截,并通过 FeignInvocationHandler 转化为 HTTP 请求。
  4. 请求通过 Client 发出,可以与 Ribbon 集成实现负载均衡。
  5. 返回结果由 Decoder 解析为对应的 Java 对象。

❤觉得有用的可以留个关注~❤

相关推荐
星河梦瑾31 分钟前
SpringBoot相关漏洞学习资料
java·经验分享·spring boot·安全
黄名富35 分钟前
Redis 附加功能(二)— 自动过期、流水线与事务及Lua脚本
java·数据库·redis·lua
love静思冥想36 分钟前
JMeter 使用详解
java·jmeter
言、雲39 分钟前
从tryLock()源码来出发,解析Redisson的重试机制和看门狗机制
java·开发语言·数据库
TT哇1 小时前
【数据结构练习题】链表与LinkedList
java·数据结构·链表
Yvemil71 小时前
《开启微服务之旅:Spring Boot 从入门到实践》(三)
java
Anna。。1 小时前
Java入门2-idea 第五章:IO流(java.io包中)
java·开发语言·intellij-idea
.生产的驴2 小时前
SpringBoot 对接第三方登录 手机号登录 手机号验证 微信小程序登录 结合Redis SaToken
java·spring boot·redis·后端·缓存·微信小程序·maven
爱上语文2 小时前
宠物管理系统:Dao层
java·开发语言·宠物
王ASC2 小时前
SpringMVC的URL组成,以及URI中对/斜杠的处理,解决IllegalStateException: Ambiguous mapping
java·mvc·springboot·web