大家不知道有没有好奇过各种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