關(guān)于接口ApplicationContext中的getBean()方法使用
我們知道可以通過ApplicationContext的getBean方法來獲取Spring容器中已初始化的bean。
getBean一共有以下四種方法原型
- 1,l getBean(String name)
- 2,l getBean(Class<T> type)
- 3,l getBean(String name,Class<T> type)
- 4,l getBean(String name,Object[] args)
下來我們分別來探討以上四種方式獲取bean的區(qū)別。
其中實(shí)體類Person定義如下:
public class Person {
?
private String name;
private int age;
public Person(){}
?
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
?
public int getAge() {
return age;
}
?
public void setAge(int age) {
this.age = age;
}
?
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
}applicationContext.xml注冊有id為p的bean,配置如下:
<bean id="p" class="com.bean.Person"> ? <property name="name" value="張三"/> ? <property name="age" value="18"/> ? </bean>
第一種:l getBean(String name)
參數(shù)name表示IOC容器中已經(jīng)實(shí)例化的bean的id或者name,且無論是id還是name都要求在IOC容器中是唯一的不能重名。
那么這種方法就是通過id或name去查找獲取bean.獲取bean的參考代碼如下:
@Test
?
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
?
Person p = (Person) ctx.getBean("p");
?
System.out.println(p);
}第二種:l getBean(Class<T> type)
參數(shù)Class<T> type表示要加載的Bean的類型。
如果該類型沒有繼承任何父類(Object類除外)和實(shí)現(xiàn)接口的話,那么要求該類型的bean在IOC容器中也必須是唯一的。
比如applicationContext.xml配置兩個(gè)類型完全一致的bean,且都沒有配置id和name屬性。
<bean class="com.bean.Person"> ? <property name="name" value="張三"/> ? <property name="age" value="18"/> ? </bean> ? <bean class="com.bean.Person"> ? <property name="name" value="李四"/> ? <property name="age" value="20"/> ? </bean>
那么通過com.bean.Person這種類型來查找bean,參考代碼如下:
@Test
public void testPerson()
?
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
?
Person p = ctx.getBean(Person.class);
?
System.out.println(p);
}但是由于屬于com.bean.Person的bean在IOC容器中不唯一,所以這里會(huì)拋出NoUniqueBeanDefinitionException異常。
由此我們可以總結(jié)getBean(String name)和getBean(Class<T> type)的異同點(diǎn)。
- 相同點(diǎn):都要求id或者name或者類型在容器中的唯一性。
- 不同點(diǎn):getBean(String name)獲得的對象需要類型轉(zhuǎn)換而getBean(Class<T> type)獲得的對象無需類型轉(zhuǎn)換。
第三種:l getBean(String name,Class<T> type)
這種方式比較適合當(dāng)類型不唯一時(shí),再通過id或者name來獲取bean。
例如applicationContext.xml配置有如下bean:
<bean id="p1" class="com.bean.Person"> ? <property name="name" value="張三"/> ? <property name="age" value="18"/> ? </bean> ? <bean name="p2" class="com.bean.Person"> ? <property name="name" value="李四"/> ? <property name="age" value="20"/> ? </bean>
參考代碼如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
?
Person p = ctx.getBean("p2",Person.class);
?
System.out.println(p);
}這樣可以獲取到名字叫”李四”的對象。
第四種:l getBean(String name,Object[] args)
這種方式本質(zhì)還是通過bean的id或者name來獲取bean,通過第二個(gè)參數(shù)Object[] args可以給bean的屬性賦值,賦值的方式有兩種:構(gòu)造方法和工廠方法。
但是通過這種方式獲取的bean必須把scope屬性設(shè)置為prototype,也就是非單例模式。
先在com.factory包下設(shè)計(jì)有如下的工廠類:
public class PersonFactory {
//靜態(tài)工廠注入
public static Person getPersonInstance(String name,int age)throws Exception
{
?
Person p = (Person)Class.forName("com.bean.Person").newInstance();
?
Method m = p.getClass().getMethod("setName", java.lang.String.class);
?
m.invoke(p, name);
?
m = p.getClass().getMethod("setAge", int.class);
?
m.invoke(p, age);
?
return p;
}
}在applicationContext.xml中配置有如下bean:
<bean name="p3" class="com.bean.Person" scope="prototype"/>
獲取bean的參考代碼如下:
@Test
?
public void testPerson()
?
{
?
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
?
Person p = (Person) ctx.getBean("p3",new Object[]{"王五",35});
?
System.out.println(p);
}如果想通過工廠注入屬性,在applicationContext.xml配置如下bean:
<bean name="p3" class="com.factory.PersonFactory" factory-method="getPersonInstance" scope="prototype"> <constructor-arg name="name"> ? <null/> ? </constructor-arg> ? <constructor-arg name="age" value="0"/> ? </bean>
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java參數(shù)傳遞及值傳遞實(shí)現(xiàn)原理詳解
這篇文章主要介紹了Java參數(shù)傳遞及值傳遞實(shí)現(xiàn)原理詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08
spring cloud學(xué)習(xí)教程之config修改配置詳解
這篇文章主要給大家介紹了關(guān)于spring cloud學(xué)習(xí)教程之config修改配置的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-09-09
java List循環(huán)與Map循環(huán)的總結(jié)
這篇文章主要介紹了java List循環(huán)與Map循環(huán)的總結(jié)的相關(guān)資料,并附代碼實(shí)例,幫助大家學(xué)習(xí)理解,需要的朋友可以參考下2016-11-11
Java實(shí)現(xiàn)導(dǎo)出pdf格式文件的示例代碼
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)導(dǎo)出pdf格式文件的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02
IDEA中編寫并運(yùn)行shell腳本的實(shí)現(xiàn)
這篇文章主要介紹了IDEA中編寫并運(yùn)行shell腳本的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
springcloud連接遠(yuǎn)程nacos失敗顯示localhost服務(wù)連接失敗的問題解決
這篇文章主要介紹了springcloud連接遠(yuǎn)程nacos失敗顯示localhost服務(wù)連接失敗的問題解決,文中有詳細(xì)的代碼示例供大家參考,對大家解決問題有一定的幫助,需要的朋友可以參考下2024-03-03
Jpa中Specification的求和sum不生效原理分析
這篇文章主要為大家介紹了Jpa中Specification的求和sum不生效原理示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08

