Feign是如何设计的,设计架构分析

本文详细介绍了Feign的设计架构,包括其基于接口的动态代理实现、注解解析、请求构建、消息编码、拦截器、日志记录、重试机制等关键步骤。此外,还讨论了Feign如何在SpringCloud中使用SpringMVC注解简化客户端开发,并分析了其性能瓶颈和默认使用的HttpURLConnection。文章最后提到了Feign的拓展实现和日志等级选项,为读者提供了深入理解Feign的全面视角。

LD is tigger forever,CG are not brothers forever, throw the pot and shine.
Modesty is not false, solid is not naive, treacherous but not deceitful, stay with good people, and stay away from poor people.
talk is cheap, show others the code,Keep progress,make a better result.

目录

概述

封装了Http调用流程,更适合面向接口化的变成习惯.
在这里插入图片描述

架构特性

设计思路

PHASE 1. 基于面向接口的动态代理方式生成实现类

interface GitHub {
  @RequestLine("GET /repos/{owner}/{repo}/contributors")
  List<Contributor> contributors(@Param("owner") String owner, @Param("repo") String repo);
}

public static class Contributor {
  String login;
  int contributions;
}

public class MyApp {
  public static void main(String... args) {
    GitHub github = Feign.builder()
                         .decoder(new GsonDecoder())
                         .target(GitHub.class, "https://api.github.com");
  
    // Fetch and print a list of the contributors to this library.
    List<Contributor> contributors = github.contributors("OpenFeign", "feign");
    for (Contributor contributor : contributors) {
      System.out.println(contributor.login + " (" + contributor.contributions + ")");
    }
  }
}

在Feign 底层,通过基于面向接口的动态代理方式生成实现类,将请求调用委托到动态代理实现类,基本原理如下所示:

PHASE 2. 根据Contract协议规则,解析接口类的注解信息,解析成内部表现:
在这里插入图片描述
默认Contract 实现
Feign 默认有一套自己的协议规范,规定了一些注解,可以映射成对应的Http请求;

上述的例子中,尝试调用GitHub.getContributors(“foo”,“myrepo”)的的时候,会转换成如下的HTTP请求:
GET /repos/foo/myrepo/contributors
HOST XXXX.XXX.XXX

注解接口Target
@RequestLine定义HttpMethod 和 UriTemplate. UriTemplate 中使用{} 包裹的表达式,可以通过在方法参数上使用@Param 自动注入
@Param定义模板变量,模板变量的值可以使用名称的方式使用模板注入解析
@Headers定义头部模板变量,使用@Param 注解提供参数值的注入。如果该注解添加在接口类上,则所有的请求都会携带对应的Header信息;如果在方法上,则只会添加到对应的方法请求上
@QueryMap定义一个键值对或者 pojo,参数值将会被转换成URL上的 query 字符串上
@HeaderMap定义一个HeaderMap, 与 UrlTemplate 和HeaderTemplate 类型,可以使用@Param 注解提供参数值
当前Spring Cloud 微服务解决方案中,为了降低学习成本,采用了Spring MVC的部分注解来完成 请求协议解析,也就是说 ,写客户端请求接口和像写服务端代码一样:客户端和服务端可以通过SDK的方式进行约定,客户端只需要引入服务端发布的SDK API,就可以使用面向接口的编码方式对接服务:
PHASE 3. 基于 RequestBean,动态生成Request

在这里插入图片描述
根据传入的Bean对象和注解信息,从中提取出相应的值,来构造Http Request 对象:
PHASE 4. 使用Encoder 将Bean转换成 Http报文正文(消息解析和转码逻辑)
在这里插入图片描述
PHASE 5. 拦截器负责对请求和返回进行装饰处理
在请求转换的过程中,Feign 抽象出来了拦截器接口,用于用户自定义对请求的操作:
比如,如果希望Http消息传递过程中被压缩,可以定义一个请求拦截器:

PHASE 6. 日志记录
在发送和接收请求的时候,Feign定义了统一的日志门面来输出日志信息 , 并且将日志的输出定义了四个等级:

级别 说明
NONE 不做任何记录
BASIC 只记录输出Http 方法名称、请求URL、返回状态码和执行时间
HEADERS 记录输出Http 方法名称、请求URL、返回状态码和执行时间 和 Header 信息
FULL 记录Request 和Response的Header,Body和一些请求元数据

PHASE 7 . 基于重试器发送HTTP请求
Feign 内置了一个重试器,当HTTP请求出现IO异常时,Feign会有一个最大尝试次数发送请求,以下是Feign核心

PHASE 8. 发送Http请求
Feign 整体框架非常小巧,在处理请求转换和消息解析的过程中,基本上没什么时间消耗。真正影响性能的,是处理Http请求的环节。
如上所述,由于默认情况下,Feign采用的是JDK的HttpURLConnection,所以整体性能并不高

实现思路分析

拓展实现

相关工具如下:

实验效果:(解决思路)

分析:

小结:

主要讲述了一些Feign里面有许多不足,请大家指正~

参考资料和推荐阅读

1.链接: 参考资料.

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

迅捷的软件产品制作专家

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值