容器是什么?spring中是如何体现的?一直有疑惑,这两天看了一下Spring管理bean的Demo,对于Spring中的容器有了简单的认识。
我们知道,容器是一个空间的概念,一般理解为可盛放物体的地方。在Spring容器通常理解为BeanFactory或者ApplicationContext。我们知道spring的IOC容器能够帮我们创建对象,对象交给spring管理之后我们就不用手动去new对象。
其中有BeanFactory与ApplicationContext两种方式可以创建对象。
那么BeanFactory与ApplicationContext的区别是什么?
采用了工厂设计模式,负责读取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就扮演了一个容器的角色。