• 二级接口ListableBeanFactory


     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 

  • 相关阅读:
    ylbASP.NET技术搭建不错的网站列表
    Arichmetic算法解析和面试经常出的算法题大总结
    Console算法递归算法示例
    ASP.NET MVC:解析 MVC+ADO.NET Entity(实体类)+Oracle
    PageDesign2010NetEase网易微博
    HTMLcolor:rgb()颜色渐进
    .NET创建虚拟表,将 SQL 表中的数据导入虚拟表
    Console算法冒泡排序法|倒水法
    WPF TranslateTransform的使用
    C# 泛型编程?
  • 原文地址:https://www.cnblogs.com/alice-cj/p/10484177.html
Copyright © 2020-2023  润新知