原文:https://blog.csdn.net/qq_23927391/article/details/80625578
我們知道可以通過(guò)ApplicationContext的getBean方法來(lái)獲取Spring容器中已初始化的bean袖迎。getBean一共有以下四種方法原型:
-
getBean(String name)
-
getBean(Class<T> type)
-
getBean(String name,Class<T> type)
-
getBean(String name,Object[] args)
下來(lái)我們分別來(lái)探討以上四種方式獲取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注冊(cè)有id為p的bean,配置如下:
<bean id="p" class="com.bean.Person">
<property name="name" value="張三"/>
<property name="age" value="18"/>
</bean>
1. getBean(String name)
參數(shù)name表示IOC容器中已經(jīng)實(shí)例化的bean的id或者name,且無(wú)論是id還是name都要求在IOC容器中是唯一的不能重名深胳。那么這種方法就是通過(guò)id或name去查找獲取bean.獲取bean的參考代碼如下:
@Test
public void testPerson()
{
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
Person p = (Person) ctx.getBean("p");
System.out.println(p);
}
2. getBean(Class<T> type)
參數(shù)Class<T> type表示要加載的Bean的類型。如果該類型沒(méi)有繼承任何父類(Object類除外)和實(shí)現(xiàn)接口的話,那么要求該類型的bean在IOC容器中也必須是唯一的。比如applicationContext.xml配置兩個(gè)類型完全一致的bean,且都沒(méi)有配置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>
那么通過(guò)com.bean.Person這種類型來(lái)查找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)獲得的對(duì)象需要類型轉(zhuǎn)換而getBean(Class<T> type)獲得的對(duì)象無(wú)需類型轉(zhuǎn)換织盼。
3. getBean(String name,Class<T> type)
這種方式比較適合當(dāng)類型不唯一時(shí)杨何,再通過(guò)id或者name來(lái)獲取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);
}
這樣可以獲取到名字叫”李四”的對(duì)象沥邻。測(cè)試結(jié)果如下:
4. getBean(String name,Object[] args)
這種方式本質(zhì)還是通過(guò)bean的id或者name來(lái)獲取bean,通過(guò)第二個(gè)參數(shù)Object[] args可以給bean的屬性賦值危虱,賦值的方式有兩種:構(gòu)造方法和工廠方法。但是通過(guò)這種方式獲取的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);
}
如果想通過(guò)工廠注入屬性,在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>