zoukankan      html  css  js  c++  java
  • Spring是如何管理Bean

      容器是什么?spring中是如何体现的?一直有疑惑,这两天看了一下Spring管理bean的Demo,对于Spring中的容器有了简单的认识。

       我们知道,容器是一个空间的概念,一般理解为可盛放物体的地方。在Spring容器通常理解为BeanFactory或者ApplicationContext。我们知道spring的IOC容器能够帮我们创建对象,对象交给spring管理之后我们就不用手动去new对象。

     其中有BeanFactory与ApplicationContext两种方式可以创建对象。

    那么BeanFactory与ApplicationContext的区别是什么?

       BeanFactory采用了工厂设计模式,负责读取bean配置文档,管理bean的加载,实例化,维护bean之间的依赖关系,负责bean的声明周期。而ApplicationContext除了提供上述BeanFactory所能提供的功能之外,还提供了更完整的框架功能:国际化支持、aop、事务等。同时BeanFactory在解析配置文件时并不会初始化对象,只有在使用对象getBean()才会对该对象进行初始化,而ApplicationContext在解析配置文件时对配置文件中的所有对象都初始化了,getBean()方法只是获取对象的过程。

    所以ApplicationContext的应用场景更广泛。

     ApplicationContext是如何管理Bean呢?下面这个Demo简单模仿了这个原理:

       1.建立一个类PersonServiceBean,并在xml文件中进行配置。

    public class PersonServiceBean implements PersonService   {  
        public void save(){  
            System.out.println("我是save()的方法");  
        }  
    }  
    <bean id="personService" class="cn.itcast.service.impl.PersonServiceBean"></bean>  

     2.建立类BeanDefinition,提供一个构造函数,将其作为每个bean的公共转型类。

    public class BeanDefinition {  
          
        private String id;    
        private String className;  
          
        public BeanDefinition(String id, String className) {  
            this.id = id;  
            this.className = className;  
        }
    //get set方法  
    }
      

      3.建立容器类tgbApplicationContext。

     1 /** 
     2  * 测试spring容器 
     3  * @author hegang 
     4  * 
     5  */  
     6 public class tgbClassPathXMLApplicationContext {  
     7       
     8     private List<BeanDefinition> beanDefines =new ArrayList<BeanDefinition>();  
     9     private Map<String,Object> sigletons =new HashMap<String,Object>();  
    10       
    11       
    12     public tgbClassPathXMLApplicationContext(String filename){  
    13         this.readXML(filename);  
    14         this.instanceBeans();  
    15     }  
    16   
    17     /** 
    18      * 完成bean的实例化 
    19      */  
    20     private void instanceBeans() {  
    21         for(BeanDefinition beanDefinition :  beanDefines){  
    22             try {  
    23                 if(beanDefinition.getClassName()!=null && !"".equals(beanDefinition.getClassName().trim())){  
    24                   sigletons.put(beanDefinition.getId(),Class.forName(beanDefinition.getClassName()).newInstance());  
    25                 }  
    26             } catch (Exception e) {  
    27                 // TODO Auto-generated catch block  
    28                 e.printStackTrace();  
    29             }  
    30               
    31         }  
    32           
    33     }  
    34   
    35     /** 
    36      * 读取xml配置文件 
    37      * @param filename 
    38      */  
    39     private void readXML(String filename) {  
    40         SAXReader saxReader = new SAXReader(); //创建读取器  
    41         Document document =null;  
    42         try{  
    43             URL xmlpath = this.getClass().getClassLoader().getResource(filename);  
    44             document =saxReader.read(xmlpath);  
    45             Map<String,String> nsMap =new HashMap<String,String>();  
    46             nsMap.put("ns","http://www.springframework.org/schema/beans"); //加入命名空间  
    47             XPath xsub = document.createXPath("//ns:beans/ns:bean"); //创建beans/bean查询路径  
    48              xsub.setNamespaceURIs(nsMap); // 设置命名空间  
    49             List<Element> beans = xsub.selectNodes(document); // 获取文档下所有的bean节点  
    50             for(Element element:beans){  
    51                 String id =element.attributeValue("id"); // 获取id属性值  
    52                 String clazz =element.attributeValue("class"); // 获取class属性值  
    53                 BeanDefinition beanDefine =new BeanDefinition(id,clazz);  
    54                 beanDefines.add(beanDefine);  
    55             }  
    56           
    57         } catch(Exception e){  
    58             e.printStackTrace();  
    59         }  
    60           
    61     }  
    62       
    63     /** 
    64      * 获取bean实例 
    65      * @param beanName 
    66      * @return 
    67      */  
    68     public Object getBean(String beanName){  
    69         return this.sigletons.get(beanName);  
    70     }  
    71 }  

    该类中拥有一个List<BeanDefinition>泛型集合类以及一个Map<String,Object>集合。通过查看代码我们知道这个容器类所做的事情如下:

        a.读取配置文件bean.xml,并根据文件中bean的id,class属性实例化一个BeanDefinition,装入泛型集合中。

        b.通过循环+反射,将List<BeanDefinition>中的bean加入到Map<String,Object>中,这里用反射将bean中的className属性转换为一个实例化的bean对象加入到了Map中。

        c.提供一个对外的接口,通过传入参数获取该bean。

      4.下面就是通过容器类获取具体bean的代码了。

    1 public class SpringTest {  
    2     @Test  
    3     public void instanceSpring(){  
    4         tgbClassPathXMLApplicationContext ctx =new tgbClassPathXMLApplicationContext("beans.xml");    
    5         PersonService personService =(PersonService) ctx.getBean("personService");  
    6         personService.save();  
    7     }  
    8 }  

    通过这样的Demo,可以清楚看到Spring容器做的事情。它在初始化的时候将配置文件中bean以及相对应关系的配置都加入到ApplicationContext,通过一系列的转换将这些bean实例化,bean被它进行了管理,所以ApplicationContext就扮演了一个容器的角色。

  • 相关阅读:
    利用Python进行数据分析-Pandas(第六部分-数据聚合与分组运算)
    利用Python进行数据分析-Pandas(第五部分-数据规整:聚合、合并和重塑)
    利用Python进行数据分析-Pandas(第四部分-数据清洗和准备)
    使用VBA从工作表中读图片,以及给工作表中写文件
    利用Python进行数据分析-Pandas(第三部分)
    利用Python进行数据分析-Pandas(第二部分)
    Shape.Type属性名称及对应值列表
    利用Python进行数据分析-Pandas(第一部分)
    下载文件
    github访问很慢的解决办法
  • 原文地址:https://www.cnblogs.com/smellpawn/p/10799441.html
Copyright © 2011-2022 走看看