添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

错误描述:

版本使用的是SpringBoot: 2.1.0.RELEASE,SpringCloud: Greenwich.M1,OpenFeign: 2.1.0.M2
报错:

The bean 'xxxx.FeignClientSpecification', defined in null, could not be registered. 
A bean with that name has already been defined in null and overriding is disabled.
Caused by: org.springframework.beans.factory.support.BeanDefinitionOverrideException: 
Invalid bean definition with name xxxx.FeignClientSpecification' defined in null: 
Cannot register bean definition [Generic bean: class [org.springframework.cloud.openfeign.FeignClientSpecification]; 
scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null;
factoryMethodName=null; initMethodName=null; destroyMethodName=null] for bean 'xxxx.FeignClientSpecification': 
There is already [Generic bean: class [org.springframework.cloud.openfeign.FeignClientSpecification]; scope=; abstract=false; lazyInit=false; 
autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false;
factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null] bound.

多个接口上的@FeignClient(“相同服务名”)会报错,overriding is disabled,即出现了相同的Bean名。

在启动类中添加的@EnableFeignClients中,可看到@import引入FeignClientsRegistrar实现注入。
在这里插入图片描述
核心是BeanDefinitionRegistry
在这里插入图片描述
注册FeignClients

    public void registerFeignClients(AnnotationMetadata metadata,
			BeanDefinitionRegistry registry) {
		ClassPathScanningCandidateComponentProvider scanner = getScanner();
		scanner.setResourceLoader(this.resourceLoader);
		Set<String> basePackages;
		Map<String, Object> attrs = metadata
				.getAnnotationAttributes(EnableFeignClients.class.getName());
		AnnotationTypeFilter annotationTypeFilter = new AnnotationTypeFilter(
				FeignClient.class);
		final Class<?>[] clients = attrs == null ? null
				: (Class<?>[]) attrs.get("clients");
		if (clients == null || clients.length == 0) {
			scanner.addIncludeFilter(annotationTypeFilter);
			basePackages = getBasePackages(metadata);
		else {
			final Set<String> clientClasses = new HashSet<>();
			basePackages = new HashSet<>();
			for (Class<?> clazz : clients) {
				basePackages.add(ClassUtils.getPackageName(clazz));
				clientClasses.add(clazz.getCanonicalName());
			AbstractClassTestingTypeFilter filter = new AbstractClassTestingTypeFilter() {
				@Override
				protected boolean match(ClassMetadata metadata) {
					String cleaned = metadata.getClassName().replaceAll("\\$", ".");
					return clientClasses.contains(cleaned);
			scanner.addIncludeFilter(
					new AllTypeFilter(Arrays.asList(filter, annotationTypeFilter)));
		for (String basePackage : basePackages) {
			Set<BeanDefinition> candidateComponents = scanner
					.findCandidateComponents(basePackage);
			for (BeanDefinition candidateComponent : candidateComponents) {
				if (candidateComponent instanceof AnnotatedBeanDefinition) {
					// verify annotated class is an interface
					AnnotatedBeanDefinition beanDefinition = (AnnotatedBeanDefinition) candidateComponent;
					AnnotationMetadata annotationMetadata = beanDefinition.getMetadata();
					Assert.isTrue(annotationMetadata.isInterface(),
							"@FeignClient can only be specified on an interface");
					Map<String, Object> attributes = annotationMetadata
							.getAnnotationAttributes(
									FeignClient.class.getCanonicalName());
					String name = getClientName(attributes);
					registerClientConfiguration(registry, name,
							attributes.get("configuration"));
					registerFeignClient(registry, annotationMetadata, attributes);

其中registerFeignClient方法需要装配BeanDefinitionBuilder对象,可以看到这里实际注入的是FeignClientFactoryBean
BeanDefinitionHolder中传递了beanName,而它是等于annotationMetadata.getClassName();,即被FeignClient注解的接口的全限定类名。
那么它的Bean名不会重复,错误原因并不是它引起的。
在这里插入图片描述
而注册configuration实际注入的是FeignClientSpecification,这就是我们所报错的already存在的Bean。
可看到 beanName:name + "." + FeignClientSpecification.class.getSimpleName()
在这里插入图片描述
而其中的name参数,就是在registerFeignClients方法中通过getClientName()获得。

String name = getClientName(attributes);
registerClientConfiguration(registry, name, attributes.get("configuration"));

getClientName()源码中,可以看到name就是@FeignClient注解中的serviceId,name,value。
beanName:name + "." + FeignClientSpecification.class.getSimpleName()
故而我们在使用相同名称的FeigClient注解时,注入到Ioc的是相同Bean名。
所以错误是由FeignClientSpecification类引起的。
在这里插入图片描述

在application.yml中配置:

spring:
  main:
    allow-bean-definition-overriding: true

在SpringBoot 2.1之前,这个配置默认就是true,而在2.1做了更改。
设置为true后,因为FeignClientSpecification的原因,FeignClient注解的configuration参数会被覆盖。

解决配置覆盖参考:
https://blog.csdn.net/neosmith/article/details/82349449

个人觉得Stackoverflow中有位大佬说的不错:
https://stackoverflow.com/questions/53103991/cant-stat-app-after-updating-springboot-from-2-0-6-release-to-2-1-0-release

别的情况下也可能会因为Bean名重复引起该错误,更佳的解决方案是尽可能避免去定义相同的Bean名。
而Feign远程调用服务因为业务需求出现在不同在接口上使用相同的服务名,将allow-bean-definition-overriding设置为true,如果在其他情况下没有重复Bean名,那么受影响的应该就只是Feign的configuration(然而configuration我都没另外做配置,覆盖也不影响- -)。

评论里大佬推荐在注解中添加 contextId 来区分,解决该问题。

@FeignClient(name="common-service", contextId = "example")
                    由FeignClientSpecification引起的A bean with that name has already been defined in null and overriding is disabled;对关于Feign的部分源码解读
				
报错展示 给产品某版本的微服务模块产品做客户化功能时,启动遇到了如下错误: Caused by: org.springframework.beans.factory.support.BeanDefinitionOverrideException: Invalid bean definition with name 'xxx.FeignClientSpecification' defined in...
Consider renaming one of the beans or enabling overriding by setting spring.main.allow-bean-definiti
报错提示 Description: The bean 'xxx', defined in null, could not be registered. A bean with that name has already been defined in null and overriding is disabled. Action: Consider renaming one of the beans or enabling overriding by setting spring.main.allow-be
The bean 'xxxxx', defined in null, could not be registered. A bean with that name has already ...
org.springframework.beans.factory.support.BeanDefinitionOverrideException: Invalid bean definition with name 'XXX.FeignClientSpecification' defined in null: Cannot register bean definition...
今儿在启动项目的时候报这个错误,一脸懵逼 我在配置文件中配置了spring.main.allow-bean-definition-overriding=true啊 为啥还会出现这个错误呢,后来经过测试,可能是复制文件导致的,把工程下的iml文件删除掉,然后重新生成就好了 The bean 'statisticsService.FeignClientSpecification', defined ...
Invalid bean definition with name 'xxx.FeignClientSpecification' defined in null: Cannot register bean definition [Generic bean: class [org.springframework.cloud.openfeign.FeignClientSpecification]; scope=; abstract=false; lazyInit=null 2、原因分析
feign.codec.DecodeException 是一个 Feign 库提供的异常类,用于表示在 Feign 客户端解码响应时发生的异常。 当 Feign 客户端尝试将响应解码为特定的数据类型时,如果解码过程中出现了异常,就会抛出 DecodeException 异常。常见的异常原因包括响应格式不正确、响应数据缺失或响应数据类型不匹配等。 在处理 Feign 客户端的响应时,需要对 DecodeException 进行捕获和处理,以便更好地了解请求的失败原因并采取适当的措施。
IDEA闪退Failed to write core dump. Minidumps are not enabled by default on client versions of Windows 37497
CSDN-Ada助手: 非常感谢CSDN博主的分享,这篇博客对于理解字节码与类的加载器的关系非常有帮助。我觉得可以继续深入探讨类的加载器的机制,尤其是在分布式系统中的应用。下一篇博客可以写关于“分布式系统中的类加载器机制”,相信会对其他用户有很大的帮助,尤其是在进行分布式应用开发时。期待您的下一篇博客! 为了方便博主创作,提高生产力,CSDN上线了AI写作助手功能,就在创作编辑器右侧哦~(https://mp.csdn.net/edit?utm_source=blog_comment_recall )诚邀您来加入测评,到此(https://activity.csdn.net/creatActivity?id=10450&utm_source=blog_comment_recall)发布测评文章即可获得「话题勋章」,同时还有机会拿定制奖牌。 WebUploader实现分块上传(断点续传)着重后端Java代码的实现 v17560043117: 因为前端上传的文件可能会太大,只能截取部分文件获取MD5,请问要校验的话,后端截取部分文件获取MD5的方式能指教一下吗,谢谢啦 Feign报错'xx.FeignClientSpecification', defined in null, could not be registered. qq_45776829: 谢谢,舒服了表情包 Non-resolvable import POM: Failure to find org.springframework.cloud:spring-cloud-dependencies:pom: ‭ᑋᵉᑊᑊᵒ ᵕ̈: IDEA抽取方法的快捷键(修改快捷键) W_chuanqi: 我的和搜狗输入法冲突