手记

SpringCloud- 第十一篇 Feign+Ribbon

1概述

由于Spring Cloud Feign的客户端负载均衡是通过Spring Cloud Ribbon实现的所以可以直接通过配置Ribbon客户端的方式来自定义各个服务客户端调用的参数

2全局配置

ribbon.ConnectTimeout=500 
ribbon.ReadTimeOut=5000

3指定服务配置

userService.ribbon.ConnectTimeout=500 
userService.ribbon.ReadTimeout=2000

4重试机制

userService.ribbon.ConnectTimeout=500
userService.ribbon.ReadTimeout=2000
userService.ribbon.OkToRetryOnAllOperations=true
userService.ribbon.MaxAutoRetriesNextServer=2
userService.ribbon.MaxAutoRetries=1

Ribbon的超时与Hystrix的超时是两个概念。一般需要让hystrix的超时时间大于Ribbon的超时时间否则Hystrix命令超时后直接熔断重试机制就没有任何意义了

5参数绑定

常用绑定参数的方式

  • @RequestParam 绑定单个请求参数值
  • @PathVariable 绑定URI模板变量值;
  • @RequestHeader 绑定请求头数据;
  • @RequestBody 绑定请求的内容区数据并能进行自动类型转换等

注意在定义各参数绑定的时候@RequestParam和@RequestHeader等可以指定参数名称的注解它们的value值千万不能少。在spring mvc程序中这些注解会根据指定参数名来作为默认值但是在fegin中绑定参数必须通过value属性来指明具体的参数名不然会抛出IllegalStateException异常value属性不能为空。

6Feign的默认配置

  • 概述
    Spring Cloud的Feign的一个中心概念就是客户端。 每个Feign客户端都是组合的组件的一部分它们一起工作以按需调用远程服务器并且该集合具有将其作为使用@FeignClient注释的参数名称。
    SpringCloud使用FeignClientsConfiguration创建一个新的集合作为每个命名客户端的ApplicationContext应用上下文这包含feign.Decoderfeign.Encoder和feign.Contract。
  • Spring Cloud Netflix默认为Feign提供以下bean
    Decoder feignDecoderResponseEntityDecoder其中包含SpringDecoder
    Encoder feignEncoderSpringEncoder
    Logger feignLoggerSlf4jLogger
    Contract feignContractSpringMvcContract
    Feign.Builder feignBuilderHystrixFeign.Builder
    Client feignClient如果Ribbon启用则为LoadBalancerFeignClient否则将使用默认的feign客户端。
    可以自定义FeignClientsConfiguration以完全控制这一系列的配置。

7自定义配置

写一个自定义配置类注意不要放到当前ComponentScan的范围下示例如

@Configuration
public class MyConf {
    @Bean
    public Contract feignContract() {
        return new feign.Contract.Default();
    }
}

定义的是new feign.Contract.Default()所有在UserService接口中只能使用Feign自己的注解url方式使用Spring MVC的注解就会报错
写好配置后通过设置@FeignClient的configuration来使用如下
@FeignClient(value = “userService”,configuration=MyConf.class)
可以为每个Feign客户端都配置自己的默认配置

8: @FeignClient

@FeignClient标签的常用属性如下

  • name(value)指定FeignClient的名称如果项目使用了Ribbonname属性会作为微服务的名称用于服务发现
  • url: url一般用于调试可以手动指定@FeignClient调用的地址
  • configuration: Feign配置类可以自定义Feign的Encoder、Decoder、LogLevel、Contract
  • fallback: 定义容错的处理类当调用远程接口失败或超时时会调用对应接口的容错逻辑fallback指定的类必须实现@FeignClient标记的接口并使用@Component注解
  • fallbackFactory: 工厂类用于生成fallback类示例通过这个属性我们可以实现每个接口通用的容错逻辑减少重复的代码
  • path: 定义当前FeignClient的统一前缀添加到Feign访问服务的访问路径上
    decode404:当发生http 404错误时如果该字段位true会调用decoder进行解码否则抛出FeignException
  • serviceId属性现在已被弃用有利于name属性。
    以前使用url属性不需要name属性。现在需要使用name

9Feign和@Primary

9.1概述

当使用Feign与Hystrix回退时在同一类型的ApplicationContext中有多个bean。这将导致@Autowired不起作用因为没有一个bean标记为主。
为了解决这个问题Spring Cloud Netflix将所有Feign实例标记为@Primary所以Spring Framework将知道要注入哪个bean。在某些情况下这可能是不可取的。要关闭此行为将@FeignClient的primary属性设置为false如

@FeignClient(name = "hello", primary = false)

10Feign的HTTP Client

Feign在默认情况下使用的是JDK原生的URLConnection发送HTTP请求没有连接池但是对每个地址会保持一个长连接即利用HTTP的persistence connection 。
可以用Apache的HTTP Client替换Feign原始的http client, 从而获取连接池、超时时间等与性能息息相关的控制能力。Spring Cloud从Brixtion.SR5版本开始支持这种替换首先在项目中声明Apache HTTP Client和feign-httpclient依赖

<dependency>
	<groupId>org.apache.httpcomponents</groupId>
	<artifactId>httpclient</artifactId>
</dependency>
<dependency>
	<groupId>com.netflix.feign</groupId>
	<artifactId>feign-httpclient</artifactId>
	<version>8.18.0</version>
</dependency>

然后在application.properties中添加

feign.httpclient.enabled=true

11: 自定义的Encode、Decode、ErrorDecode

  • Feign将方法签名中方法参数对象序列化为请求参数放到HTTP请求中的过程是由编码器(Encoder)完成的。同理将HTTP响应数据反序列化为java对象是由解码器(Decoder)完成的。
  • 默认情况下Feign会将标有@RequestParam注解的参数转换成字符串添加到URL中将没有注解的参数通过Jackson转换成json放到请求体中。
  • 在Spring Cloud环境下Feign的Encoder只会用来编码没有添加注解的参数。如果你自定义了Encoder, 那么只有在编码没有注解的参数时才会调用你的Encoder。
  • 对于Decoder, 默认会委托给SpringMVC中的MappingJackson2HttpMessageConverter类进行解码。只有当状态码不在200 ~ 300之间时ErrorDecoder才会被调用。ErrorDecoder的作用是可以根据HTTP响应信息返回一个异常该异常可以在调用Feign接口的地方被捕获到。我们目前就通过ErrorDecoder来使Feign接口抛出业务异常以供调用者处理。

12: FeignClient工作过程

FeignClient相当于Spring Cloud中的RPC大致实现的过程如下

1首先通过@EnableFeignCleints注解开启FeignCleint
2根据Feign的规则实现接口并加@FeignCleint注解
3程序启动后会进行包扫描扫描所有的@FeignCleint的注解的类并将这些信息注入到IoC容器中
4当接口的方法被调用通过JDK的代理来生成具体的RequestTemplate
5RequestTemplate再生成调用的Request
6Request交给Client去处理其中Client可以是HttpUrlConnection、HttpClient也可以是Okhttp
7最后Client被封装到LoadBalanceClient类这个类结合类Ribbon做到了负载均衡真正发送请求出去

13特性继承

13.1概述

当使用SpringMVC的注解来绑定服务接口时候几乎可以完全从服务提供方的Controller中复制操作构建出相应的服务客户端绑定接口。既然存在这么多复制操作我们自然需要考虑这部分内容是否可以得到进一步的抽象。Spring Cloud Feign中针对该问题提供了继承特性来帮助解决这些复制操作以进一步减少编码量。

13.2基本方法

  • 就是把接口提出来把model提出来然后provider和client都使用这个工程
  • Provider这边在Controller中不再包含以往会定义的映射注解@RequestMapping而参数的注解定义在重写的时候自动带过来了这个类中除了要实现接口逻辑之外只需要增加了@RestController注解使该类成为一个REST接口类。
  • Consumer这边直接注入Feign标注的接口完全像调用本地接口一样。

13.3注意

  • Feign接口只能支持一层继承而且不能多继承
  • 官方不建议在服务器和客户端之间共享接口因为它引入了紧耦合并且实际上并不适用于当前形式的Spring MVC方法参数映射不被继承。

14Hystrix配置

14.1: 概述

Spring Cloud Feign中集成了HystrixSpring Cloud Feign客户端的方法都封装到Hystrix命令中进行服务保护。

14.2:全局配置

直接使用它的默认配置前缀hystrix.command.default就可以进行设置比如设置全局的超时时间 Hystrix默认的超时时间是1秒
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds=5000 在对hystrix进行设置之前需要确认Feign的Hystrix功能是开启的。

14.3: 禁用hystrix

可以通过设置feign.hystrix.enabled为false或者使用hystrix.command.default.execution.timeout.enabled=false来关闭熔断功能
如果不想全局地关闭Hystrix支持而只想针对某个服务客户端关闭Hystrix支持时需要通过使用@Scope(“protototype”)注解为指定的客户端配置Feign.Builder实例如

@Configuration
public class DisableHystrixConfigutation {
@Bean
@Scope("prototype")
public Feign.Builder feignBuilder() {
	return Feign.builder();
}}

在该服务的Feign接口中引入该配置

@FeignClient(value = "user-service",configuration = DisableHystrixConfigutation.class) ...

14.4: 指定命令配置

在实际应用中可能会根据实际业务情况制定出不同的配置方案可以采用hystrix.command.作为前缀。而默认情况下会采用feign客户端中的方法名作为标识。需要注意的是由于方法名有可能会重复这个时候相同的方法名的hystrix配置会共用所以在进行方法与配置的时候需要做好一定的规划。

14.5: 服务降级配置

Spring Cloud Feign在定义服务客户端的时候与Spring cloud Ribbon有很大的差别由于HystrixCommand定义被封装起来无法通过@HystrixCommand注解的fallback参数那样来指定具体的服务降级处理方法。Spring Cloud Feign提供了另外一种简单的方式。
方法大致是定义一个Feign客户端的服务降级类UserServiceFallback实现UserService接口其中每个重写方法的实现逻辑都可以用来定义相应的服务降级逻辑在服务绑定接口中通过@FeignClient注解的fallback属性来指定对应的服务降级实现类。

14.6: 请求压缩

Spring Cloud Feign支持对请求与响应进行GZIP压缩以减少通信过程中的性能损耗只需要通过下面的两个参数设置就能开启请求与相应的压缩功能

feign.compression.request.enabled=true 
feign.compression.response.enabled=true

同时还可以对请求压缩做一些更细致的设置比如下面的配置内容指定压缩的请求数据类型并设置了请求压缩的大小下限只有超过这个大小的请求才会进行压缩

feign.compression.request.enabled=true
feign.compression.request.mime-types=text/xml,application/xml,application/json
feign.compression.request.min-request-size=2048

15日志配置

Spring Cloud Feign在构建被@FeignClient注解修饰的服务客户端时会为每一个客户端都创建一个feign.Logger实例可以配置logging.level.的参数配置格式来开启指定feign客户端的debug日志其中为feign客户端定义接口的完整路径例如

logging.level.com.cc.springcloud_feign_api.UserService2=debug

注意只添加该配置还无法实现对debug日志的输出。这是因为feign客户端默认的Logger.Level对象定义为NONE级别该级别不会记录任何Feign调用过程中的信息所以需要调整它的级别针对全局的日志级别可以在应用主类中加入

@Bean
Logger.Level feignLoggerLevel(){
    return Logger.Level.FULL;
}

也可以实现配置类然后在Feign客户端来指定配置类以实现不同的日志级别。
对于Feign的Logger级别主要有下面4类可根据实际需要进行调整使用

  • none不记录任何信息
  • basic仅记录请求方法url以及响应状态码和执行时间
  • headers除了记录basic级别的信息之外还会记录请求和响应的头信息。
  • FULL记录所有请求与响应的明细包括头信息请求体元数据等。
1人推荐
随时随地看视频
慕课网APP