Feign源码分析

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之后,有个注入属性方法,里面有一处后置处理器方法InstantiationAwareBeanPostProcessorpostProcessPropertyValues,改方法的目的是在把属性值应用到真正的字段前做一些动作(我的另外一篇专门关于后置处理器的文章)。

最后兜兜转转会到这个方法里面,此时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请求

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值