Java漫谈程序员

浅析Feign代理bean 注入原理

2019-03-15  本文已影响115人  聊聊其他

大家不知道有没有好奇过各种spring框架中的神奇功能是怎么实现的,例如Feign 的接口,和Jpa 的接口,Spring 是如何驱动实现的呢。下面我来分析一下

在了解FeignClient 注解中发现,FeignClient 注解的接口都是Proxy 类。那到底是如何生成的呢?

spring在注入bean的时候,会根据容器中bean 的类型或者name 来匹配bean 的注入,
但是interface类型的接口是无法作为bean 生成的,所以必须有一个机制来生成实现了对应接口的类,然后注入到Ioc容器中。

FeignClient 中使用到的是BeanFactory 来生成对应的bean实体,下面来看下Feign中的源码:

首先我们跟踪@EnableFeignClients注解, 里面可以看到有一个FeignClientsRegistrar配置类

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(FeignClientsRegistrar.class)
public @interface EnableFeignClients {
    ....
}

这个类实现了ImportBeanDefinitionRegistrar 接口,这个接口有什么特别呢?在spring中,如果bean实现了这个接口,那么这个bean会被对待为配置Bean ,而不作为注入的bean生成。

继续跟踪代码:

@Override
public void registerBeanDefinitions(AnnotationMetadata metadata,
        BeanDefinitionRegistry registry) {
    registerDefaultConfiguration(metadata, registry);
    registerFeignClients(metadata, registry);
}

这个就是Bean 生成的入口,一直跟踪代码下面,会有个方法:registerFeignClient,这里可以看到一个很关键的地方

private void registerFeignClient(BeanDefinitionRegistry registry,
        AnnotationMetadata annotationMetadata, Map<String, Object> attributes) {
        ...
BeanDefinitionBuilder definition = BeanDefinitionBuilder
            .genericBeanDefinition(FeignClientFactoryBean.class);
            
            ...
        }

这里是获得bean的工厂类,然后根据工厂类来生成proxy。

跟踪到FeignClientFactoryBean,可以发现工厂类实现类FactoryBean<T>接口
这里重点关注getObject()方法,Srping Ioc容器就是根据这个方法来获取对应的Bean,那么还有一个问题,前面刚刚提到,spring是可以根据name 或者类型来注入bean的,这里应该还需要指定对应的type 才能让spring决定注入的bean,方法就是 FactoryBean 接口的 Class<?> getObjectType()

而FeignClientFactoryBean中实现的方法如下:

@Override
public Class<?> getObjectType() {
    return this.type;
}

从配置中获取到type,这里我们往回看,可以看到:

private void registerFeignClient(BeanDefinitionRegistry registry,
        AnnotationMetadata annotationMetadata, Map<String, Object> attributes) {
    String className = annotationMetadata.getClassName();
    BeanDefinitionBuilder definition = BeanDefinitionBuilder
            .genericBeanDefinition(FeignClientFactoryBean.class);
    ...
    definition.addPropertyValue("type", className);
    ...
    definition.setAutowireMode(AbstractBeanDefinition.AUTOWIRE_BY_TYPE);

这里获取的就是@FeignClient 注解上的接口类型,所以spring就能根据这个返回生成的代理对象注入到接口当中。对于想了解具体的代理生成方式的同学,可以看getObject中的详细实现。

这里就分享得差不多了,有问题请留言。笔者技术水平有限,如有出错,欢迎指出,谢谢。

技术交流,可以加微信:youliaodao_zc

上一篇 下一篇

猜你喜欢

热点阅读