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 对象。

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

相关推荐
解孔明21 分钟前
IDEA2023.1添加java虚拟机启动参数,打开断言
java·开发语言
关关不烦恼23 分钟前
【Java数据结构】二叉树
java·开发语言·数据结构
苹果酱056724 分钟前
使用 React Testing Library 测试自定义 React Hooks
java·开发语言·spring boot·后端·中间件
好奇的菜鸟33 分钟前
Java技术体系:深入理解JDK与JRE及其产品线
java·开发语言
api7741 分钟前
1688商品详情API返回值中的售后保障与服务信息
java·服务器·前端·javascript·python·spring·pygame
疑惑的杰瑞1 小时前
[乱码]确保命令行窗口与主流集成开发环境(IDE)统一采用UTF-8编码,以规避乱码问题
java·c++·vscode·python·eclipse·sublime text·visual studio
自身就是太阳1 小时前
深入理解 Spring 事务管理及其配置
java·开发语言·数据库·spring
喵手1 小时前
Java零基础-多态详解
java·开发语言·python
麋鹿会飞但不飘1 小时前
EasyExcel拿表头(二级表头)爬坑,invokeHeadMap方法
java·spring boot·excel
Gauss松鼠会1 小时前
GaussDB关键技术原理:高弹性(四)
java·大数据·网络·数据库·分布式·gaussdb