JAVASpring

Springboot 注入 不同包中类名相同冲突问题解决

问题:


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”)。

解决方法三:


将其中一个实现类改为不同的名字。