之前使用RestTemplate进行服务之间的调用,在代码风格上来看不太统一和美观,存在
代码可读性差,编程体验不统一
参数复杂URL难以维护
的问题,Feign就可以解决这个问题。
Feign是一个声明式的http客户端,它使得调用远程服务就像调用本地服务一样简单,只需要创建一个接口并添加一个注解即可。
其作用就是帮助我们优雅的实现http请求的发送,解决上面提到的问题。
而且Feign默认集成了Ribbon,所以使用Feign默认就实现了负载均衡的效果。
我们在消费者这一方,也就是调用其他服务的这一方进行Feign的配置
编码风格瞬间统一,看着也舒服多了。
那feign是怎么实现负载均衡的呢?其实其内部已经自带了Ribbon。如图:
Feign的自定义配置:
Feign可以支持很多的自定义配置,如下表所示:
类型 | 作用 | 说明 |
---|---|---|
feign.Logger.Level | 修改日志级别 | 包含四种不同的级别:NONE(默认)、BASIC、HEADERS、FULL NONE ,不记录。默认选项 BASIC ,仅记录请求方法和URL以及响应状态代码和执行时间。 HEADERS ,记录基本信息以及请求和响应标头。 FULL ,记录请求和响应的标题,正文和元数据。 |
feign.codec.Decoder | 响应结果的解析器 | http远程调用的结果做解析,例如解析json字符串为java对象 |
feign.codec.Encoder | 请求参数编码 | 将请求参数编码,便于通过http请求发送,例如POST请求,将请求参数编码到请求体中 |
feign.Contract | 支持的注解格式 | 默认是SpringMVC的注解 |
feign.Retryer | 失败重试机制 | 请求失败的重试机制,默认是没有,不过会使用Ribbon的重试,例如A服务无法访问,会尝试访问集群中的B服务 |
一般情况下,默认值就能满足我们使用,如果要自定义时,只需要创建自定义的@Bean覆盖默认Bean即可。
那么下面就以修改日志级别为例,展示自定义配置、
这里我们用Apache的HttpClient来演示
① 在order-service的pom文件中引入Apache的HttpClient依赖
- <!--httpClient的依赖内置连接池 -->
- <dependency>
- <groupId>io.github.openfeign</groupId>
- <artifactId>feign-httpclient</artifactId>
- </dependency>
② 在order-service的application.yml中添加配置
- feign:
- httpclient:
- enabled: true # 开启feign对HttpClient的支持
- max-connections: 200 # 设置最大的连接数
- max-connections-per-route: 50 # 并行接收一个服务的请求数量
优化总结:使用连接池:
最优实践:由于提供这种可能要多次调用同一个消费者,那么就会出现相同的代码写很多次,为了避免这种代码的重复,就要想办法进行优化。
方法一,就是将相同的代码抽取成一个接口,然后被消费者继承。但是这样的话实际上的代码量并没有减少很多,优化效果不好,所以官方不推荐。虽然不推荐,但是遵循了面向契约的编程思想,所以企业中用户的还是比较多的。优点,面向契约,缺点,耦合度高!
方法二,将Feign抽取出来,消费者只需要将Feign模块添加到依赖中,就可以使用全部的方法。
优点是,耦合度低,缺点是,如果只想用其中的一个功能,但是还是不得不将全部功能都引入进来。
但是,方法二的实现过程中,会把FeignClient抽取到别的模块,但是这样会导致消费者的启动类无法扫描到FeignClient,那么就无法将FeignClient这个类注入到spring容器当中。那么解决办法有两种:
两者选其一,只要想办法让spring扫描到这个类,注入到Spring容器当中,问题自然就迎刃而解了。