1 package org.springframework.beans.factory; 2 3 import java.lang.annotation.Annotation; 4 import java.util.Map; 5 6 import org.springframework.beans.BeansException; 7 import org.springframework.core.ResolvableType; 8 9 //可将Bean逐一列出的工厂 10 public interface ListableBeanFactory extends BeanFactory { 11 12 //是否存在给定的bean名称的信息 13 boolean containsBeanDefinition(String beanName); 14 15 //BeanDefinition的总数 16 int getBeanDefinitionCount(); 17 18 19 //返回工程中所有bean的名字 20 String[] getBeanDefinitionNames(); 21 22 23 //返回对于指定类型的beanName 24 String[] getBeanNamesForType(ResolvableType type); 25 26 //返回指定类型bean的名字 27 String[] getBeanNamesForType(Class<?> type); 28 29 30 /* 31 * 返回指定类型的名字 32 * includeNonSingletons为false表示只取单例Bean,true则不是 33 * allowEagerInit为true表示立刻加载,false表示延迟加载。 34 * 注意:FactoryBeans都是立刻加载的。 35 */ 36 String[] getBeanNamesForType(Class<?> type, boolean includeNonSingletons, boolean allowEagerInit); 37 38 39 //根据bean的类型返回bean和bean的Map集合 40 <T> Map<String, T> getBeansOfType(Class<T> type) throws BeansException; 41 42 43 <T> Map<String, T> getBeansOfType(Class<T> type, boolean includeNonSingletons, boolean allowEagerInit) 44 throws BeansException; 45 46 47 // 根据注解类型,查找所有有这个注解的Bean名和Bean的Map 48 String[] getBeanNamesForAnnotation(Class<? extends Annotation> annotationType); 49 50 51 // 根据注解类型,查找所有有这个注解的Bean名和Bean的Map 52 Map<String, Object> getBeansWithAnnotation(Class<? extends Annotation> annotationType) throws BeansException; 53 54 55 //根据指定Bean名和注解类型查找指定的Bean 56 <A extends Annotation> A findAnnotationOnBean(String beanName, Class<A> annotationType) 57 throws NoSuchBeanDefinitionException; 58 59 }
这个工厂中扩展了
3个 BeanDefinition有关的接口,
3个根据指定类型返回beanName的数组
2个根据指定类型返回bean和存储Bean的Map集合
3个和注解有关的返回bean和beanMap的接口方法
对于BeanDefinition中包含了一个类在Spring工厂中所有属性。
这个工厂如同名字一样,可以返回所有bean的实例,但是这个工厂并没有直接返回实例的方法,只有通过条件返回指定bean的Name数组,Map等,
其实不需要直接返回实例的方法,因为我们拿到了bean的所有名字,我们就可以使用getbean,父接口中得方法,也就是BeanFactory进行获取Bean的实例!
可以通过ApplicationContext的getBean方法来获取Spring容器中已初始化的bean。getBean一共有以下四种方法原型:
1 getBean(String name) 2 3 getBean(Class<T> type) 4 5 getBean(String name,Class<T> type) 6 7 getBean(String name,Object[] args)
下来我们分别来探讨以上四种方式获取bean的区别。
其中实体类Person定义如下:
1 public class Person { 2 3 private String name; 4 private int age; 5 6 public Person(){} 7 8 public Person(String name, int age) { 9 this.name = name; 10 this.age = age; 11 } 12 13 public String getName() { 14 return name; 15 } 16 17 public void setName(String name) { 18 this.name = name; 19 } 20 21 public int getAge() { 22 return age; 23 } 24 25 public void setAge(int age) { 26 this.age = age; 27 } 28 29 @Override 30 public String toString() { 31 return "Person [name=" + name + ", age=" + age + "]"; 32 } 33 }
applicationContext.xml注册有id为p的bean,配置如下:
1 <bean id="p" class="com.bean.Person"> 2 <property name="name" value="张三"/> 3 <property name="age" value="18"/> 4 </bean>
第一种方法:getBean(String name)
参数name表示IOC容器中已经实例化的bean的id或者name,且无论是id还是name都要求在IOC容器中是唯一的不能重名。那么这种方法就是通过id或name去查找获取bean.获取bean的参考代码如下:
1 @Test 2 public void testPerson() 3 { 4 ApplicationContext ctx = new 5 ClassPathXmlApplicationContext("applicationContext.xml"); 6 7 Person p = (Person) ctx.getBean("p"); 8 System.out.println(p); 9 10 }
第二种方法:getBean(Class<T> type)
参数Class<T> type表示要加载的Bean的类型。如果该类型没有继承任何父类(Object类除外)和实现接口的话,那么要求该类型的bean在IOC容器中也必须是唯一的。比如applicationContext.xml配置两个类型完全一致的bean,且都没有配置id和name属性。
1 <bean class="com.bean.Person"> 2 <property name="name" value="张三"/> 3 <property name="age" value="18"/> 4 </bean> 5 <bean class="com.bean.Person"> 6 <property name="name" value="李四"/> 7 <property name="age" value="20"/> 8 </bean>
那么通过com.bean.Person这种类型来查找bean,参考代码如下:
1 @Test 2 public void testPerson() 3 { 4 ApplicationContext ctx = new 5 ClassPathXmlApplicationContext("applicationContext.xml"); 6 Person p = ctx.getBean(Person.class); 7 System.out.println(p); 8 9 }
但是由于属于com.bean.Person的bean在IOC容器中不唯一,所以这里会抛出NoUniqueBeanDefinitionException异常。
由此我们可以总结getBean(String name)和getBean(Class<T> type)的异同点。
相同点:都要求id或者name或者类型在容器中的唯一性。
不同点:getBean(String name)获得的对象需要类型转换而getBean(Class<T> type)获得的对象无需类型转换。
第三种方法:getBean(String name,Class<T> type)
这种方式比较适合当类型不唯一时,再通过id或者name来获取bean。
1 <bean id="p1" class="com.bean.Person"> 2 <property name="name" value="张三"/> 3 <property name="age" value="18"/> 4 </bean> 5 <bean name="p2" class="com.bean.Person"> 6 <property name="name" value="李四"/> 7 <property name="age" value="20"/> 8 </bean>
1 @Test 2 public void testPerson() 3 { 4 ApplicationContext ctx = new 5 ClassPathXmlApplicationContext("applicationContext.xml"); 6 Person p = ctx.getBean("p2",Person.class); 7 System.out.println(p); 8 }
这样可以获取到名字叫”李四”的对象。
第四种方法:getBean(String name,Object[] args)
这种方式本质还是通过bean的id或者name来获取bean,通过第二个参数Object[] args可以给bean的属性赋值,赋值的方式有两种:构造方法和工厂方法。但是通过这种方式获取的bean必须把scope属性设置为prototype,也就是非单例模式。
先在com.factory包下设计有如下的工厂类:
1 public class PersonFactory { 2 //静态工厂注入 3 public static Person getPersonInstance(String name,int age)throws Exception 4 { 5 Person p = (Person)Class.forName("com.bean.Person").newInstance(); 6 Method m = p.getClass().getMethod("setName", java.lang.String.class); 7 m.invoke(p, name); 8 m = p.getClass().getMethod("setAge", int.class); 9 m.invoke(p, age); 10 return p; 11 } 12 }
在applicationContext.xml中配置有如下bean:
<bean name="p3" class="com.bean.Person" scope="prototype"/>
获取bean的参考代码如下:
1 @Test 2 public void testPerson() 3 { 4 ApplicationContext ctx = new 5 ClassPathXmlApplicationContext("applicationContext.xml"); 6 Person p = (Person) ctx.getBean("p3",new Object[]{"王五",35}); 7 System.out.println(p); 8 }
如果想通过工厂注入属性,在applicationContext.xml配置如下bean:
1 <bean name="p3" class="com.factory.PersonFactory" factory-method="getPersonInstance" scope="prototype"> 2 <constructor-arg name="name"> 3 <null/> 4 </constructor-arg> 5 <constructor-arg name="age" value="0"/> 6 </bean>
---------------------
参考链接:
https://www.cnblogs.com/java-synchronized/p/6777556.html
https://blog.csdn.net/qq_23927391/article/details/80625578