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

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

相关推荐
菜鸟蹦迪27 分钟前
八股文实战之JUC:ArrayList不安全性
java
2501_9032386527 分钟前
Spring MVC配置与自定义的深度解析
java·spring·mvc·个人开发
逻各斯38 分钟前
redis中的Lua脚本,redis的事务机制
java·redis·lua
计算机毕设指导640 分钟前
基于Springboot学生宿舍水电信息管理系统【附源码】
java·spring boot·后端·mysql·spring·tomcat·maven
计算机-秋大田1 小时前
基于Spring Boot的兴顺物流管理系统设计与实现(LW+源码+讲解)
java·vue.js·spring boot·后端·spring·课程设计
计算机小白一个1 小时前
蓝桥杯 Java B 组之背包问题、最长递增子序列(LIS)
java·数据结构·蓝桥杯
二十雨辰1 小时前
[Java基础]网络编程
java·开发语言
ACGkaka_1 小时前
抓包工具(三)Wireshark代理抓包Java程序的HTTPS请求
java·https·wireshark
Struggle Sheep2 小时前
容器化部署tomcat
java·tomcat
大腕先生2 小时前
微服务环境搭建&架构介绍(附超清图解&源代码)
微服务·云原生·架构