对于 feign,无疑就是整合了 ribbon + hystrix,然后封装为对开发使用起来更加友好的 jar。因此本质上,feign 是 ribbon + hystrix. 因此,在学习 Feign 之前,建议先把 ribbon, hystrix 学习一下。
@EnableFeignClient 注解导入了 FeignClientsRegistrar 类, FeignClientsRegistrar 实现了 ImportBeanDefinitionRegistrar 接口,完成了对 @FeignClient 类的加载。
为每个 @FeignClient 注入的 Feign 配置类 FeignClientFactoryBean,实现了 FactoryBean。在 getObject()时,创建 ReflectiveFeign.FeignInvocationHandler。 FeignInvocationHandler 对应的就是被 @FeignClient 注解标注的类。方法则对应 SynchronousMethodHandler。 当在调用时 Feign 方法时,会被 FeignInvocationHandler 代理。
FeignInvocationHandler 创建流程如下
Feign 动态代理执行流程如下
当 feign.hystrix.enable=true 时,此时会被 @FeignClient 会被 HystrixInvocationHandler 代理,先走 Hystrix 的逻辑,然后再走 SynchronousMethodHandler 的逻辑。
网上大部分的关于 openFeign 文件上传的文章,可能都会教你这么做:
1、引入对应的jar
<dependency> <groupId>io.github.openfeign.form</groupId> <artifactId>feign-form</artifactId> <version>3.8.0</version> </dependency>
实际上,这个不需要再引入,因为 openFeign 已经引入了该依赖
@Configuration public class ServiceClientConfig { @Autowired private ObjectFactory<HttpMessageConverters> messageConverters; @Bean public Encoder feignFormEncoder() { return new SpringFormEncoder(new SpringEncoder(messageConverters)); } }
实际上,SpringFormEncoder 不需要再装入,openFeign 已经装入该 Encoder。
@PostMapping(value = "/upload", consumes = MediaType.MULTIPART_FORM_DATA_VALUE) String upload(@RequestPart(name = "file") MultipartFile file);