@Primary注解


在spring 中使用注解,常使用@Autowired, 默認是根據類型Type來自動注入的。但有些特殊情況,對同一個接口,可能會有幾種不同的實現類,而默認只會采取其中一種的情況下 @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 {
        private static final Logger logger = LoggerFactory.getLogger(SingerService.class);
        @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 根據類型無法選擇到底注入哪一個。這個時候@Primay 可以閃亮登場了。

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

如果代碼改成這樣,再次運行,結果如下: 
“I am singing in Bocelli voice: song lyrics”, 用@Primary 告訴spring 在猶豫的時候優先選擇哪一個具體的實現。

二、用@Qualifier這個注解來解決問題

將上面的兩個類改為如下:

@Component // 加注解,讓spring識別
@Qualifier("metalSinger")
public class MetalSinger implements Singer{
 
    @Override
    public String sing(String lyrics) {
        return "I am singing with DIO voice: "+lyrics;
    }
}
 
@Component
@Qualifier("opreaSinger")
public class OperaSinger implements Singer {
    @Override
    public String sing(String lyrics) {
        return "I am singing in Bocelli voice: "+lyrics;
    }
}
    @Component
    public class SingerService {
        private static final Logger logger = LoggerFactory.getLogger(SingerService.class);
     
        @Autowired
        private Singer singer;
     
        @Qualifier("opreaSinger")
        public String sing(){
            return singer.sing("song lyrics");
        }
    }

 

@Configuration把一個類作為一個IoC容器,它的某個方法頭上如果注冊了@Bean,就會作為這個Spring容器中的Bean。 
@Scope注解 作用域 
@Lazy(true) 表示延遲初始化 
@Service用於標注業務層組件、 
@Controller用於標注控制層組件(如struts中的action) 
@Repository用於標注數據訪問組件,即DAO組件。 
@Component泛指組件,當組件不好歸類的時候,我們可以使用這個注解進行標注。 
@Scope用於指定scope作用域的(用在類上) 
@PostConstruct用於指定初始化方法(用在方法上) 
@PreDestory用於指定銷毀方法(用在方法上) 
@Resource 默認按名稱裝配,當找不到與名稱匹配的bean才會按類型裝配。 
@DependsOn:定義Bean初始化及銷毀時的順序 
@Primary:自動裝配時當出現多個Bean候選者時,被注解為@Primary的Bean將作為首選者,否則將拋出異常 
@Autowired 默認按類型裝配,如果我們想使用按名稱裝配,可以結合@Qualifier注解一起使用 
@Autowired @Qualifier(“personDaoBean”) 存在多個實例配合使用

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM