报错说明
BeanCreationException: Error creating bean with name 'com.xxx.XxxClient':
Lookup method resolution failed;
nested exception is java.lang.IllegalStateException: Failed to introspect Class [org.springframework.cloud.openfeign.FeignClientFactoryBean] from ClassLoader [sun.misc.Launcher$AppClassLoader]
可见原因在于 FeignClientFactoryBean introspect 失败
内省(Introspector)是Java语言对Bean类属性、事件的一种缺省处理方法
内省又称作运行时类型检查
类似反射,但反射会破坏Bean的属性而内省遵从约定
代码场景
package com.xxx;
@FeignClient(value = "vxxx",contextId ="idxxx")
public interface XxxClient extends XxxFeignApi {
}
启动类扫描
@EnableFeignClients(basePackages = {"com.xxx"})
排查思路
- 查看 BeanDefinition是否创建,即是否扫描到FeignClient类
- 查看 FactoryBean 是否实例化
- 查看 FactoryBean 的 getObject 方法是否执行,即是否实例化 Bean
是否扫描
排查入口 run 方法
run 方法中重点关注
this.refreshContext(context);
委托执行
this.refresh(context);
继续委托
((AbstractApplicationContext)applicationContext).refresh();
AbstractApplicationContext extends DefaultResourceLoader implements ConfigurableApplicationContext
调用 AbstractApplicationContext 的 refresh 方法
其模板方法定义中调用 this.invokeBeanFactoryPostProcessors(beanFactory);
此处是spring的扩展点
初始化中进入 PostProcessorRegistrationDelegate 类
最后委托 ConfigurationClassPostProcessor 类,其 parse 方法
metadata = doProcessConfigurationClass(configClass, metadata);
执行 doProcessConfigurationClass 方法
其中会执行 processImports 方法导入各种import类
invokeBeanFactoryPostProcessors
排查扫描类 FeignClientsRegistrar
@Import(FeignClientsRegistrar.class)
public @interface EnableFeignClients {...}
通过import注入扫描类
class FeignClientsRegistrar
implements ImportBeanDefinitionRegistrar, ResourceLoaderAware, EnvironmentAware {
// 主要通过钩子方法将扫描的 feignClient 生成 BeanDefinition
// 通过 ImportBeanDefinitionRegistrar 的 registerBeanDefinitions 方法加入到 BeanDefinitionRegistry
@Override
public void registerBeanDefinitions(AnnotationMetadata metadata,
BeanDefinitionRegistry registry) {
registerDefaultConfiguration(metadata, registry);
// 此处加入 BeanDefinition
registerFeignClients(metadata, registry);
}
}
ImportBeanDefinitionRegistrar 接口
spring的扩展点,支持自定义逻辑封装 BeanDefinition对象;实现此接口的类会回调postProcessBeanDefinitionRegistry方法
扫描的 feignClient 注册时都是放入的 FactoryBean
解决方案:统一调用方和被调用方 openfeign 依赖包版本
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
<version>xxx</version>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-openfeign-core</artifactId>
<version>xxx</version>
</dependency>