feign主要围绕注解@FeignClient和@Autowired来分析就好了,@FeignClient是注解在一个接口上面的,这个注解是一个标志,相应的后置处理器把有该标志的类加入到bean defenition中。@Autowired是注入属性的,feign采用的是FactoryBean
的方式进行创建的。下面开始详细分析
一、注解FeignClient
在使用feign的时候都要的配置类上加上EnableFeignClients
,这个注解上面有个Import注解,这个很重要
@Import(FeignClientsRegistrar.class)
public @interface EnableFeignClients {
...................
}
Registrar
是一个加载bean定义的常见形式,他给开发者暴露出registry,让开发者自由的加载bean
熟悉spring的都知道,这些类的加载都是靠ConfigclassPostProcessor
来完成的,我简略的梳理一下流程
invokeBeanFactoryPostProcessors(beanFactory); invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry); loadBeanDefinitionsForConfigurationClass(configClass, trackedConditionEvaluator) loadBeanDefinitionsFromRegistrars(configClass.getImportBeanDefinitionRegistrars()); registerBeanDefinitions(); ```
registerBeanDefinitions
方法就是开发者可以自由进行注册bean定义的方法,spring会给你registry作为参数,feign利用的是
AnnotationTypeFilter annotationTypeFilter = new AnnotationTypeFilter(FeignClient.class);
就是利用类上是否有相应的注解,如果有就加入进来。这些其实度不是feign的东西,都是spring的
二、动态代理创建feign服务
feign利用的是@Autowired来注入的。在调用创建bean之后,有个注入属性方法,里面有一处后置处理器方法InstantiationAwareBeanPostProcessor
的postProcessPropertyValues
,改方法的目的是在把属性值应用到真正的字段前做一些动作(我的另外一篇专门关于后置处理器的文章)。
最后兜兜转转会到这个方法里面,此时beanName就是注解autowired的先关属性,requiredType就是autowired标注的那个类,这相当于又走回getBean()
这个老朋友身上
public Object resolveCandidate(String beanName, Class<?> requiredType, BeanFactory beanFactory)
throws BeansException {
return beanFactory.getBean(beanName, requiredType);
}
在getBean
的时候我惊讶的发现,改bean的class竟然是FeignClientFactoryBean
,原来在FeignClientsRegistar
中,创建beanDefinition的过程中被hardcode的指定了class
BeanDefinitionBuilder definition = BeanDefinitionBuilder.genericBeanDefinition(FeignClientFactoryBean.class);
2.1 FeignClientFactoryBean
首先他继承FactoryBean
,他的核心方法是getObject()
。在方法中主要做了构建LoaderBalancerFeignClient,当然如何和hystrix
集合还会构建HystrixTarget,然后根据是否有 fallbackFactory,进行一系列的构建。
LoaderBalancerFeignClient
的核心内容我贴出来了,他和ribbon的RibbonLoadBalancerClient其实做的事情是一样的,主要就是找出真正的LoadBalancer,在这里就是FeignLoadBalancer
(ribbon的是ZoneAwareLoadBalancer)
URI asUri = URI.create(request.url());
String clientName = asUri.getHost();
URI uriWithoutHost = cleanUrl(request.url(), clientName);
FeignLoadBalancer.RibbonRequest ribbonRequest = new FeignLoadBalancer.RibbonRequest(
this.delegate, request, uriWithoutHost);
IClientConfig requestConfig = getClientConfig(options, clientName);
return lbClient(clientName).executeWithLoadBalancer(ribbonRequest,
requestConfig).toResponse();
下面的基本和ribbon相同,就是解析正服务url,进行http调用
三、Ribbon和Feign
ribbon更像是对restTempalte的扩展,而Feign更符合面向对象的思想,但是最终他们实现的细节确实相同的,都是构建出相应client,然后找到队形的LoadBalancer,然后进行解析,执行http请求