spring中向一個單例bean中注入非單例bean的方法詳解
前言
看到這個題目相信很多小伙伴都是懵懵的,平時我們的做法大都是下面的操作
@Component
public class People{
@Autowired
private Man man;
}
這里如果Man是單例的,這種寫法是沒有問題的,但如果Man是原型的,這樣是否會存在問題。
錯誤實例演示
這里有一個原型(生命周期為prototype)的類
package com.example.myDemo.component;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
@Component
@Scope(value = "prototype")
public class Man {
public void eat() {
System.out.println("I like beef");
}
}
有一個單例(生命周期為singleton)的類
package com.example.myDemo.component;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.stereotype.Component;
@Component
public class Woman {
//使用依賴注入的方式,注入原型的Man @Autowired
private Man man;
public void eat() {
System.out.println("man:"+man);
System.out.println("I like fruits");
}
}
下面看測試方法,
package com.example.myDemo;
import com.example.myDemo.component.MyFactoryBean;
import com.example.myDemo.component.Woman;
import com.example.myDemo.po.Student;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration;
import org.springframework.context.ApplicationContext;
@SpringBootApplication(exclude={DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class})
public class MyDemoApplication {
public static void main(String[] args) {
ApplicationContext ac=SpringApplication.run(MyDemoApplication.class, args);
Woman woman=(Woman)ac.getBean("woman");
for(int i=0;i<5;i++){
woman.eat();
}
}
}
看下測試結(jié)果,

上面的結(jié)果顯示W(wǎng)oman中的man是單例的,因為5次循環(huán)打印打出的結(jié)果是同一個對象,發(fā)生了什么,
Woman是單例的,Man是原型的,我們使用常規(guī)的@Autowired注解注入的卻是同一個實例,這里想下為什么Man是一個對象,Woman是單例的,意味著在整個spring容器中只有一個實例,在屬性注入的時候肯定也只會注入一次,所以其中Man屬性也只能是一個實例,出現(xiàn)上圖的結(jié)果也就不稀奇了。
現(xiàn)在有這樣一個需求要向單例bean中注入原型bean,要怎么實現(xiàn)這樣的需求
實現(xiàn)ApplicationContextAware接口
都知道ApplicationContextAware接口是spring提供的一個擴展點,實現(xiàn)該接口的類可以獲得ApplicationContext
Woamn類改成下面的樣子
package com.example.myDemo.component;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public class Woman implements ApplicationContextAware {
private Man man;
private ApplicationContext ac;
public void eat() {
this.man = (Man) ac.getBean("man");
System.out.println("man:" + man);
System.out.println("I like fruits");
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.ac = applicationContext;
}
}
Woman實現(xiàn)了ApplicationContextAware接口,注入了ApplicaitonContext對象,然后再eat()方法中通過AppicationContext獲得Man的實例,看測試結(jié)果,

可以看到man屬性是多例的也就是符合原型模式的定義。
思考下為什么采用這種方式可以達到注入原型bean的目的
在eat()方法中使用ApplicationContext的getBean方法獲取Man,eat()方法每執(zhí)行一次均會調(diào)用一次getBean方法,getbean方法在執(zhí)行的時候的時候會判斷Man的生命周期,如果是原型(prototype)的,那么每調(diào)用一次就會重新實例化一個Man,所以會出現(xiàn)上述的結(jié)果。
該方法有一個很大的缺點那就是和spring耦合度太高,不符合降低系統(tǒng)的耦合度的要求。
lookup method
spring也考慮了向一個單例bean中注入原型bean的情況,提供了@Lookup注解,在XML配置方式下是<lookup-method>標簽,這里僅使用注解的方式演示,
Woman類修改如下,
package com.example.myDemo.component;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public class Woman {
private Man man;
public void eat() {
this.man = createMan();
System.out.println("man:" + man);
System.out.println("I like fruits");
}
@Lookup
public Man createMan(){
return null;
}
}
看下測試結(jié)果,

上圖顯示man是一個多例的,也就是向單例bean中注入了原型bean,其作用的是@Lookup注解。
通過@Lookup注解便完成了注入原型bean的目的,留個思考問題spring是如何做到的?
lookup method簽名
被@Lookup注解或<lookup-method>配置的方法有如下要求,
public|protected [abstract] return-type methodName(no-argments)
- 方法可以是public也可以是protected;
- 方法可以是抽象的也可以是非抽象的;
- 方法的返回值是要注入的類型,這里是prototype類型的類;
- 方法沒有入?yún)ⅲ?/li>
- 方法體可以是空的。具體返回值可以是null或任何類型,對結(jié)果沒有影響;
總結(jié)
分享了向單例bean中注入原型bean的方式,使用lookup的方式會更簡潔些。
這還可能是道面試題哦,各位小伙伴注意嘍。lookup的原理下次分享,敬請關(guān)注
到此這篇關(guān)于spring中向一個單例bean中注入非單例bean的文章就介紹到這了,更多相關(guān)spring注入非單例bean內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring MVC如何設(shè)置請求頭和響應(yīng)頭的Header
這篇文章主要介紹了Spring MVC如何設(shè)置請求頭和響應(yīng)頭的Header問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2025-03-03
使用Jenkins一鍵打包部署SpringBoot項目的步驟詳解
任何簡單操作的背后,都有一套相當(dāng)復(fù)雜的機制,本文將以SpringBoot應(yīng)用的在Docker環(huán)境下的打包部署為例,詳細講解如何使用Jenkins一鍵打包部署SpringBoot應(yīng)用,文中通過圖文結(jié)合講解的非常詳細,需要的朋友可以參考下2023-11-11
使用MyBatis攔截器實現(xiàn)sql查詢權(quán)限動態(tài)修改代碼實例
這篇文章主要介紹了使用MyBatis攔截器實現(xiàn)sql查詢權(quán)限動態(tài)修改代碼實例,為了不耦合,現(xiàn)在的方案是在需要鑒權(quán)的Mybatis?Mapper方法上增加一個注解,在運行過程中判斷該注解存在即對sql進行修改,需要的朋友可以參考下2023-08-08
java將一個整數(shù)轉(zhuǎn)化成二進制代碼示例
這篇文章主要介紹了java將一個整數(shù)轉(zhuǎn)化成二進制代碼示例,具有一定借鑒價值,需要的朋友可以參考下2017-12-12
使用Java注解和反射實現(xiàn)JSON字段自動重命名
這篇文章主要介紹了如何使用Java注解和反射實現(xiàn)JSON字段自動重命名,文中通過代碼示例和圖文介紹的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-08-08
java 中用split分割字符串,最后的空格等不被拆分的方法
下面小編就為大家?guī)硪黄猨ava 中用split分割字符串,最后的空格等不被拆分的方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-02-02

