快速入门

之前,我们利用Nacos实现了服务的治理,利用RestTemplate实现了服务的远程调用。但是远程调用的代码太复杂了:

【OpenFeign】-连接池、最佳实践_远程调用

而且这种调用方式,与原本的本地方法调用差异太大,编程时的体验也不统一,一会儿远程调用,一会儿本地调用。

因此,我们必须想办法改变远程调用的开发模式,让远程调用像本地方法调用一样简单。而这就要用到OpenFeign组件了。

OpenFeign是一个声 明式的htp客户端,是SpringCloud在Eureka公 司开源的Feign基础.上改造而来。官方地址: https://github.com/OpenFeign/feign

其作用就是基于SpringMVC的常见注解,帮我们优雅的实现http请求的发送。

【OpenFeign】-连接池、最佳实践_连接池_02

OpenFeign已经被SpringCloud自动装配,实现起来非常简单

引入依赖,包括OpenFeign和负载均衡组件SpringCloudLoadBalancer

【OpenFeign】-连接池、最佳实践_微服务_03

通过@EnableFeignClients注解,启用OpenFeign功能

【OpenFeign】-连接池、最佳实践_连接池_04

 编写FeignClient

【OpenFeign】-连接池、最佳实践_微服务_05

@FeignClient("item-service")
public interface ItemClient {

    @GetMapping("items")
    List<ItemDTO> queryItemByIds(@RequestParam("ids") Collection<Long> ids);

}

使用FeignClient,实现远程调用

【OpenFeign】-连接池、最佳实践_远程调用_06

【OpenFeign】-连接池、最佳实践_微服务_07

【OpenFeign】-连接池、最佳实践_远程调用_08

【OpenFeign】-连接池、最佳实践_连接池_09

连接池

Feign底层发起http请求,依赖于其它的框架。其底层支持的http客户端实现包括:

  • HttpURLConnection:默认实现,不支持连接池
  • Apache HttpClient :支持连接池
  • OKHttp:支持连接池

OpenFeign整合OKHttp的步骤如下:

引入依赖

【OpenFeign】-连接池、最佳实践_微服务_10

开启连接池功能

【OpenFeign】-连接池、最佳实践_微服务_11

最佳实践

将来我们要把与下单有关的业务抽取为一个独立微服务:trade-service,不过我们先来看一下hm-service中原本与下单有关的业务逻辑。

入口在com.hmall.controller.OrderControllercreateOrder方法,然后调用了IOrderService中的createOrder方法。

由于下单时前端提交了商品id,为了计算订单总价,需要查询商品信息:

【OpenFeign】-连接池、最佳实践_远程调用_12

也就是说,如果拆分了交易微服务(trade-service),它也需要远程调用item-service中的根据id批量查询商品功能。这个需求与cart-service中是一样的。

因此,我们就需要在trade-service中再次定义ItemClient接口,这不是重复编码吗? 有什么办法能加避免重复编码呢?

相信大家都能想到,避免重复编码的办法就是抽取。不过这里有两种抽取思路:

  • 思路1:抽取到微服务之外的公共module
  • 思路2:每个微服务自己抽取一个module

【OpenFeign】-连接池、最佳实践_连接池_13

方案1抽取更加简单,工程结构也比较清晰,但缺点是整个项目耦合度偏高。

方案2抽取相对麻烦,工程结构相对更复杂,但服务之间耦合度降低。

日志

OpenFeign只会在FeignClient所在包的日志级别为DEBUG时,才会输出日志。而且其日志级别有4级:

  • NONE:不记录任何日志信息,这是默认值。
  • BASIC:仅记录请求的方法,URL以及响应状态码和执行时间
  • HEADERS:在BASIC的基础上,额外记录了请求和响应的头信息
  • FULL:记录所有请求和响应的明细,包括头信息、请求体、元数据。

Feign默认的日志级别就是NONE,所以默认我们看不到请求日志。

在hm-api模块下新建一个配置类,定义Feign的日志级别:

package com.hmall.api.config;

import feign.Logger;
import org.springframework.context.annotation.Bean;

public class DefaultFeignConfig {
    @Bean
    public Logger.Level feignLogLevel(){
        return Logger.Level.FULL;
    }
}

接下来,要让日志级别生效,还需要配置这个类。有两种方式:

  • 局部生效:在某个FeignClient中配置,只对当前FeignClient生效
@FeignClient(value = "item-service", configuration = DefaultFeignConfig.class)
  • 全局生效:在@EnableFeignClients中配置,针对所有FeignClient生效。
@EnableFeignClients(defaultConfiguration = DefaultFeignConfig.class)

【OpenFeign】-连接池、最佳实践_微服务_14