Feign 详解

6 篇文章 0 订阅
1 篇文章 0 订阅

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

  • 0
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Feign是一个HTTP客户端,它可以将HTTP请求转发到其他微服务。使用Feign可以使得微服务之间的调用更加简单和优雅。下面是使用Feign的详细步骤: 1. 添加依赖 在使用Feign之前,需要先在pom.xml文件中添加Feign的依赖,如下所示: ```xml <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-openfeign</artifactId> </dependency> ``` 2. 创建Feign接口 在使用Feign时,需要先创建一个接口,用于定义需要调用的其他微服务的API。这个接口的方法签名和被调用的微服务的API方法签名必须一致。例如,如果需要调用另一个微服务的getUserInfo方法,那么Feign接口的定义如下所示: ```java @FeignClient(name = "user-service") public interface UserService { @GetMapping("/user/getUserInfo") public UserInfo getUserInfo(@RequestParam("userId") String userId); } ``` 在这个例子中,@FeignClient注解用于指定需要调用的微服务的名称,name属性的值为"user-service",这个值需要和被调用的微服务的spring.application.name属性的值一致。接着,定义了一个getUserInfo方法,这个方法的签名与被调用的微服务的getUserInfo方法的签名一致,使用@GetMapping注解标注请求的路径,这里的路径为"/user/getUserInfo"。最后,定义了一个UserInfo类型的返回值,用于封装被调用的微服务的返回结果。 3. 注入Feign接口 在需要使用Feign调用其他微服务的地方,可以直接注入Feign接口,例如在Service层中注入上面定义的UserService接口: ```java @Service public class UserServiceImpl implements UserService { @Autowired private UserService userService; public UserInfo getUserInfo(String userId) { return userService.getUserInfo(userId); } } ``` 在这个例子中,UserServiceImpl类注入了UserService接口,并且调用了getUserInfo方法。在实际运行时,Feign会根据接口的定义动态生成一个HTTP客户端,并将请求转发到其他微服务。 4. 启用Feign 在使用Feign时,需要在Spring Boot应用程序的启动类上添加@EnableFeignClients注解,如下所示: ```java @SpringBootApplication @EnableFeignClients public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` 在这个例子中,@EnableFeignClients注解用于启用Feign客户端。在启用Feign之后,Spring Boot会自动扫描所有标注了@FeignClient注解的接口,并为它们动态生成HTTP客户端。 以上就是使用Feign的详细步骤。使用Feign可以使得微服务之间的调用更加简单和优雅,提高代码的复用性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值