Feign 详解

2023-05-16

1、Feign 是什么

Feign是一个http请求调用的轻量级框架,可以以Java接口注解的方式调用Http请求。Feign通过处理注解,将请求模板化,当实际调用的时候,传入参数,根据参数再应用到请求上,进而转化成真正的请求,封装了http调用流程。

2、为什么选择 Feign

如果不使用rpc框架,那么调用服务需要走http的话,无论是使用 JDK 自带的 URLConnection,还是使用Http工具包 Apache 的httpclient, 亦或是 OkHttp, 都需要自行配置请求headbody,然后才能发起请求。获得响应体后,还需解析等操作,十分繁琐。

Feign 只需要定义一个接口,并且通过注解的形式定义好请求模板,就可以项使用本地接口一样,使用Http请求。

3、Feign 是怎么工作

Feign 是通过定义接口,并且在接口方法上使用注解定义请求模板,然后通过 Feign.builder() 进行构建后,即可像使用本地接口方法调用Http请求。简单实例如下:

// 定义接口
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");

    // 发送请求
    List<Contributor> contributors = github.contributors("OpenFeign", "feign");
    for (Contributor contributor : contributors) {
      System.out.println(contributor.login + " (" + contributor.contributions + ")");
    }
  }
}

4、Feign 使用详解

接口注解

注解目标用法
@RequestLine方法定义请求 的HttpMethod。用大括号括起来的Expression使用它们对应的带注释的参数来解析。
@Param参数定义一个模板变量,其值将用于解析相应的模板Expression,按作为注释值提供的名称提供
@Headers方法、接口定义一个请求头模板,其中可以使用大括号括起来的表达式,将使用 @Param 注解的参数解析,标注在方法上只针对某个请求,标注在类上,表示作用的所有的请求上
@QueryMap参数定义Map名称-值对或 POJO,以扩展为查询字符串。
@HeaderMap参数定义一个Map名称-值对,展开成 请求头
@Body方法类似于一个 URI 模板,他使用 @Param 注解的参数来解析模板中的表达式

模板和表达式

Feign 是由 URI 模板 Expressions 定义的简单字符串表达式,并且使用 @Param 注解的方法参数,进行解析。

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

如上所示代码中 通过 @RequestLine 注解标注的为 URL模板,其中,有大括号括起来的 wonerrepo 在发送请求时,会被有 @Param 注解的参数 ownerrepo 所替换。

表达式必须用大括号括起来,{}并且可以包含正则表达式模式,用冒号分隔: 以限制解析值。 示例 owner必须是字母。{owner:[a-zA-Z]*}

请求参数扩展

RequestLineQueryMap模板遵循URI模板 规范,该规范指定以下内容:

  • 未解析的表达式被省略。
  • 所有文字和变量值都经过 pct-encoded 编码,如果尚未encoded通过@Param注释编码或标记。

未定义或空值

未定义的表达式是指表达式的值是显式null或未提供值的表达式。根据URI 模板 - RFC 6570,可以为表达式提供空值。Feign 解析表达式时,首先判断该值是否已定义,如果已定义则查询参数将保留。如果表达式未定义,则删除查询参数。

空字符串
// 定义接口
@RequestLine("POST /user/map")
String map(@QueryMap Map<String, Object> queryMap);

Map<String, Object> queryMap = new LinkedHashMap<>();
 queryMap.put("param", "");

// 调用服务
this.client.map(queryMap);

// 解析后的路径为
http://localhost/user/map?param
没有参数
// 定义接口
@RequestLine("POST /user/map")
String map(@QueryMap Map<String, Object> queryMap);

Map<String, Object> queryMap = new LinkedHashMap<>();

// 调用服务
this.client.map(queryMap);

// 解析后的路径为
http://localhost/user/map
未定义
// 定义接口
@RequestLine("POST /user/map")
String map(@QueryMap Map<String, Object> queryMap);

Map<String, Object> queryMap = new LinkedHashMap<>();
 queryMap.put("param", null);

// 调用服务
this.client.map(queryMap);

// 解析后的路径为
http://localhost/user/map?param

请求头扩展

Feign 中可以通过 HeadersHeaderMap 两个注解来扩展请求头,并且遵循以下规则,

  • 未解析的表达式被忽略,如果请求头的值为空,这删除整个请求头。
  • 不执行pct-encoded 编码
Headers

Headers 注解可以标注到 api 方法上,也可以标注到客户端即接口上,标注在接口上,表示对所有的请求都起作用,标注在 方法上 只对所标注的方法起作用。

@Headers("Accept: application/json")  // 此处标注,表示对所有的请求都起作用
interface BaseApi<V> {
  @Headers("Content-Type: application/json") // 只对当前请求起作用
  @RequestLine("PUT /api/{key}")
  void put(@Param("key") String key, V value);
}

在方法上标注时可以设置动态的内容,如下所示:

public interface Api {
   @RequestLine("POST /")
    // 动态指定 Token 值,在方法参数中需要存在 @Param 标注的名为 token的参数
   @Headers("X-Ping: {token}") 
   void post(@Param("token") String token);
}
HeaderMap

Headers 虽然也能动态设置头信息,但是,当请求头的键和个数不确定时,Headers 就不能满足了,此时我们可以使用 HeaderMap 注解的 方法参数来更灵活的动态指定请求头

public interface Api {
   @RequestLine("POST /")
   void post(@HeaderMap Map<String, Object> headerMap);
}

请求正文扩展

Body 模板遵循与请求参数扩展相同的扩展,但有一下更改

  • 未解析的表达式被省略
  • 扩展值在放置在正文之前不会被 Encoder 进行编码
  • Content-Type 请求头必须设置

如下所示:

interface LoginClient {

  @RequestLine("POST /")
  @Headers("Content-Type: application/xml")
  @Body("<login \"user_name\"=\"{user_name}\" \"password\"=\"{password}\"/>")
  void xml(@Param("user_name") String user, @Param("password") String password);

  @RequestLine("POST /")
  @Headers("Content-Type: application/json")
  // json 花括号必须转义
  @Body("%7B\"user_name\": \"{user_name}\", \"password\": \"{password}\"%7D")
  void json(@Param("user_name") String user, @Param("password") String password);
}

public class Example {
  public static void main(String[] args) {
      // <login "user_name"="denominator" "password"="secret"/>
    client.xml("denominator", "secret"); 
       // {"user_name": "denominator", "password": "secret"}
    client.json("denominator", "secret");
  }
}

编码器

将请求正文发送到服务器的最简单方法是定义一个POST方法,该方法具有Stringorbyte[]参数而没有任何注释。您可能需要添加Content-Type标题。

interface LoginClient {
  @RequestLine("POST /")
  @Headers("Content-Type: application/json")
  void login(String content);
}

public class Example {
  public static void main(String[] args) {
    client.login("{\"user_name\": \"denominator\", \"password\": \"secret\"}");
  }
}

通过配置Encoder,您可以发送类型安全的请求正文。feign-gson这是使用扩展的示例:

static class Credentials {
  final String user_name;
  final String password;

  Credentials(String user_name, String password) {
    this.user_name = user_name;
    this.password = password;
  }
}

interface LoginClient {
  @RequestLine("POST /")
  void login(Credentials creds);
}

public class Example {
  public static void main(String[] args) {
    LoginClient client = Feign.builder()
                              .encoder(new GsonEncoder()) // 引入GsonEncoder编码器
                              .target(LoginClient.class, "https://foo.com");

    client.login(new Credentials("denominator", "secret"));
  }
}

解码器

在实际开发中,服务端返回的数据可能是个JSON字符串或者字节数组,在 Feign 中可以通过指定解码器,把响应数据解析为你想要的数据类型。以下是使用feign-gson 进行解码的实例:

public interface WebFeign {
    
    @RequestLine("POST /user/body")
    @Headers({"Content-Type: application/json"})
    User postBody(User user);
}


WebFeign webFeign = Feign.builder()
    .encoder(new GsonEncoder())
    // 指定解码器
    .decoder(new GsonDecoder())
    .target(WebFeign.class, "http://localhost:10010");

User user = new User().setName("张三")
    .setAge(20);

User body = webFeign.postBody(user);

请求客户端扩展

Feign 底层默认使用的是 JDK 自带的 URLConnection 实现的网络请求。在Feign 中也可以在构建时指定自定的底层网络请求工具,比如常用的OkHttpApache HttpClient 等。Feign 也已经实现了 这两个客户端,只需要引入依赖就可以直接使用。

<!-- 引入 HttpClient  -->
<dependency>
    <groupId>io.github.openfeign</groupId>
    <artifactId>feign-httpclient</artifactId>
    <version>${feign.version}</version>
</dependency>
<!-- 引入 OkHttp -->
<dependency>
    <groupId>io.github.openfeign</groupId>
    <artifactId>feign-okhttp</artifactId>
    <version>${feign.version}</version>
</dependency>

根据自己的需求引入依赖即可,引入依赖后在构建时指定所需的客户端即可,如下:

Feign.builder()
    .client(new ApacheHttpClient()) // 使用 HttpClient
    .logger(new Slf4jLogger())
    .decoder(new StringDecoder())
    .encoder(new GsonEncoder())
    .target(WebFeign.class, "http://localhost:10010");


Feign.builder()
    .client(new OkHttpClient()) // 使用 OkHttp
    .logger(new Slf4jLogger())
    .decoder(new StringDecoder())
    .encoder(new GsonEncoder())
    .target(WebFeign.class, "http://localhost:10010");

5、合约扩展

Feign中 定义接口时使用的注解为 @RequestLine 其格式为 HttpMethod Path, 是把请求类型和请求路径放在了一起。如果熟悉 JAX-RS 注解或者 SpringMVC 注解的开发者,可能不是太友好,值得庆幸的是,Feign 提供了扩展点,可以自定义注解解析处理类,即 Contract 接口,并且 也针对 JAX-RS 已经实现了扩展, Spring 也对 Feign 提供了扩展,即spring-cloud-openfeign。 这里先简单介绍系 JAX-RS 的用法。

引入依赖

 <dependency>
     <groupId>io.github.openfeign</groupId>
     <artifactId>feign-jaxrs</artifactId>
     <version>${feign.version}</version> <!-- 根据实际情况修改版本号 -->
</dependency>

使用也相当简单

WebFeign webFeign = Feign.builder()
    .contract(new JAXRSContract())  // 使用 JAX-RS 注解处理
    .client(new ApacheHttpClient())
    .target(WebFeign.class, "http://localhost:10011");

6、总结

Feign 是一个很好的框架工具,把繁琐的 Http 请求,抽象为以接口加注解的方式实现,也使开发者很好的面向接口编程。在目前微服务盛行的当下,Spring 也对 Feign 进行了封装,即OpenFeign ,并且相当流行。这里把最底层、最基础的Feign的用法梳理一下,能够更好的理解 Spring 封装的 OpenFeign

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Feign 详解 的相关文章

随机推荐

  • keil 提示internal command error和no sw device

    1 使用keil烧录软件的时候 xff0c jlink stlink无法识别到芯片 需要排查的问题 1 xff09 换条线 2 xff09 是不是有程序禁用了Seral Wire xff1a 使用cubeide cubeMX xff0c 容
  • 多线程编程模式之Single Threaded Execution 模式

    一 Single Threaded Execution 模式介绍 简单的来说 xff0c Single threaded execution 模式描述了在一种多线程环境下各个线程对于公用资源的使用方式 任一时刻 xff0c 只有一个线程可以
  • NVIDIA Jetson TX2 查看系统相关+运行demo

    1 查看Jetson TX2 L4T版本 xff1a head n 1 etc nv tegra release 2 查看系统版本 xff1a cat etc lsb release 3 查看系统内核 xff1a uname a 4 查看内
  • Docker镜像迁移至新的服务器(全部数据)

    1 找到你想移动的 Docker 容器的 ID 2 提交你的变更 xff0c 并且把容器保存成镜像 xff0c 命名为 newimage docker commit span class token number 3 span a09b25
  • 配置VNC环境在windows主机访问阿里云linux服务器

    配置VNC环境在windows主机访问阿里云linux服务器 虽然作为服务器使用更多的是使用字符终端连接服务器 xff0c 进行操作 xff0c 因为图形界面很消耗性能和资源 xff0c 但有的时候使用图形界面进行操作更为便捷 xff0c
  • pythondataframe输出小结

    在使用dataframe时遇到datafram在列太多的情况下总是自动换行显示的情况 xff0c 导致数据阅读困难 xff0c 效果如下 xff1a coding utf 8 import numpy as np import pandas
  • 聊聊 Redis 为什么构建自己的简单动态字符串 SDS

    我们知道 xff0c Redis 支持字符串 哈希 列表 集合和有序集合五种基本类型 那么我们如何把图片 音频 视频或者压缩文件等二进制数据保存到 Redis 中呢 xff1f 之前在使用 Memcached 缓存这类数据时是把它们转换成
  • 聊聊 Redis 高可用之持久化AOF和RDB分析

    Redis 持久化概述 Redis 是内存数据库 xff0c 数据都是存储在内存中 xff0c 为了避免进程退出导致数据的永久丢失 xff0c 需要定期将 Redis 中的数据以某种形式把内存中的数据保存到磁盘中 xff1b 当 Redis
  • mysqldump: Got error: 1044: Access denied for user XXXX when doing LOCK TABLES

    一 报错信息 在使用mysqldump 执行远程备份数据库的时候报如下错误 xff1a mysqldump Got error span class token number 1044 span Access denied span cla
  • jmap -heap [pid]运行报:Error attaching to process: sun.jvm.hotspot.debugger.DebuggerException(不允许的操作)

    一 运行环境 操作系统 xff1a Ubuntu 5 4 0 6 Java版本 xff1a JDK8 二 执行命令 jmap heap span class token punctuation span pid号 span class to
  • chkconfig: command not found

    问题描述 在 ubuntu1 16 04 10 执行 chkconfig 命令报 chkconfig command not found 说明此服务上没有安装 chkconfig 执行如下命令进行安装 span class token fu
  • Docker 基础篇 之 安装

    一 Docker安装 查看 CentOS 内核版本 Docker 要求 CentOS 系统的内核版本高于3 10 执行如下命令查询 内核版本 span class token function uname span r span class
  • Java 基础 之 Valid 验证

    一 64 Valid 简介 Bean Validation 内置的校验器 校验器说明 64 Null被注解的元素必须为 null 64 NotNull被注解的元素必须不为 null 64 AssertTrue被注解的元素必须为 true 6
  • HttpURLConnection链接详解

    HttpURLConnection链接详解 一 简介 简单来说 xff0c HttpURLConnection 是 Java 提供的发起 HTTP 请求的基础类库 xff0c 提供了 HTTP 请求的基本功能 xff0c 不过封装的比较少
  • Apache HttpClient 详解

    1 简介 HttpClient 是 Apache Jakarta Common 下的子项目 xff0c 用来提供高效的 最新的 功能丰富的支持 HTTP 协议的客户端编程工具包 xff0c 并且它支持 HTTP 协议最新的版本和建议 Htt
  • OKHttp使用详解

    1 简介 OkHttp 是一个默认高效的 HTTP 客户端 xff1a HTTP 2 支持允许对同一主机的所有请求共享一个套接字 连接池减少了请求延迟 xff08 如果 HTTP 2 不可用 xff09 透明 GZIP 缩小了下载大小 响应
  • python二维码生成与扫码

    1 import qrcode img 61 qrcode make 34 hello world 34 img get image show img save 39 hello png 39 2 import qrcode qr 61 q
  • C语言可变参数(从stdarg.h到应用)

    1 什么是可变参数函数 在C语言编程中有时会遇到一些参数可变的函数 xff0c 例如printf scanf xff0c 其函数原型为 xff1a span class token keyword int span span class t
  • OkHttp 缓存实战

    1 简介 在实际业务中可能某些查询数据 xff0c 不经常变化 xff0c 为了节省流量 提高响应速度和增强用户体验等 xff0c 把变化频率小的数据缓存到本地 xff0c 以实现复用 OkHttp 的缓存功能使用起来也比较简单和灵活 xf
  • Feign 详解

    1 Feign 是什么 Feign是一个http请求调用的轻量级框架 xff0c 可以以Java接口注解的方式调用Http请求 Feign通过处理注解 xff0c 将请求模板化 xff0c 当实际调用的时候 xff0c 传入参数 xff0c