问题:
springboot 在启动时候,常启动不起来,检查发现是不同包下面有同名的service和serviceImpl,按理说不同包下是可以有同名的类存在的,但是启动就是启动不了,报错:
org.springframework.context.annotation.ConflictingBeanDefinitionException: Annotation-specified bean name 'roleServiceImpl' for bean class [com.example.service.RoleServiceImpl] conflicts with existing, non-compatible bean definition of same name and class [com.example.roleService.RoleServiceImpl]
意思是:以Bean名字‘roleServiceImpl’注解的类[com.example.service.RoleServiceImpl]与存在的不相容的同名类[com.example.roleService.RoleServiceImpl]相冲突。
原来是在这两个实现类上面都只用了@service这个注解,根据映射规则,这两个service都映射为了roleServiceImpl,发生了冲突。
出现冲突原因:spring提供两种beanName生成策略,基于注解的sprong-boot默认使用的是AnnotationBeanNameGenerator,它生成beanName的策略就是,取当前类名(不是全限定类名)作为beanName。由此,如果出现不同包结构下同样的类名称,肯定会出现冲突。
解决方法一:
1.重新定义beanName生成策略,继承AnnotationBeanNameGenerator,重写generateBeanName,如下所示:
public class UniqueNameGenerator extends AnnotationBeanNameGenerator { @Override public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) { //如果有设置了value,则用value,如果没有则是用全类名 if (definition instanceof AnnotatedBeanDefinition) { String beanName = determineBeanNameFromAnnotation((AnnotatedBeanDefinition) definition); if (StringUtils.hasText(beanName)) { // Explicit bean name found. return beanName; }else{ //全限定类名 beanName = definition.getBeanClassName(); return beanName; } } // 使用默认类名 return buildDefaultBeanName(definition, registry); } }
2.在启动类上加注解@ComponentScan(nameGenerator = UniqueNameGenerator.class)使刚才我们自定义的BeanName生成策略生效。
@SpringBootApplication @ComponentScan(nameGenerator = UniqueNameGenerator.class) public class TestApplication { public static void main(String[] args) { SpringApplication.run(TestApplication.class, args); } }
解决方法二:
将其中一个注解变更为一个name为非roleServiceImpl的注解@service(name=”aaaa”)。
解决方法三:
将其中一个实现类改为不同的名字。