JAVA基础-注解-primary

1,502 阅读1分钟

对同一个接口,可能会有几种不同的实现类,而默认只会采取其中一种的情况下就可使用 @Primary 。 的作用就出来了。


public interface Singer {
    String sing(String lyrics);
}

@Component // 加注解,让spring识别
public class MetalSinger implements Singer{

    @Override
    public String sing(String lyrics) {
        return "I am singing with DIO voice: "+lyrics;
    }
}

public class OperaSinger implements Singer {
    @Override
    public String sing(String lyrics) {
        return "I am singing in Bocelli voice: "+lyrics;
    }
}

调用:
@Component
public class SingerService {

    @Autowired
    private Singer singer;

    public String sing(){
        return singer.sing("song lyrics");
    }
}

结果为I am singing with DIO voice: song lyrics. 原因很简单,就是 OperaSinger 这个类上面根本没有加上注解@Copmonent 或者 @Service, 所以spring 注入的时候,只能找到 MetalSinger 这个实现类. 所以才有这个结果。

但是如果一旦 OperaSinger 这个类加上了@Copmonent 或者 @Service注解,有趣的事情就会发生,你会发现一个错误的结果或异常: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [main.service.Singer] is defined: expected single matching bean but found 2: metalSinger,operaSinger 提示很明确了,spring 根据类型无法选择到底注入哪一个。

给OperaSinger 加上@Primary注解,再次运行,结果:"I am singing in Bocelli voice: song lyrics", 用@Primary 告诉spring 在犹豫的时候优先选择哪一个具体的实现。