(本文转自 ......忘了)
Bean工厂 (com.springframework.beans.factory.BeanFactory)是Spring框架最核心的接口,它提供了高级IoC的配置机制。BeanFactory使管理不同类型的Java对象成为可能,应用上下文(com.springframework.context.ApplicationContext)建立在BeanFactory基础之上,提供了更多面向应用的功能,它提供了国际化支持和框架事件体系,更易于创建实际应用。我们一般称BeanFactory为IoC容器,而称 ApplicationContext为应用上下文。但有时为了行文方便,我们也将ApplicationContext称为Spring容器。
对于两者的用途,我们可以进行简单划分:BeanFactory是Spring框架的基础设 施,面向Spring本身;ApplicationContext面向使用Spring框架的开发a者,几乎所有的应用场合我们都直接使用ApplicationContext而非底层的BeanFactory。
Spring为BeanFactory提供了多种实现,最常用的是 XmlBeanFactory。 BeanFactory接口位于类结构树的顶端,它最主要的方法就是getBean(String beanName),该方法从容器中返回特定名称的Bean,BeanFactory的功能通过其他的接口得到不断扩展。
代码示例:
beans.xml:Car的配置文件
01 | <?xml version="1.0" encoding="UTF-8" ?> | |
02 | <beans xmlns="http://www.springframework.org/schema/beans" |
03 | xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | |
04 | xmlns:p="http://www.springframework.org/schema/p" |
05 | xsi:schemaLocation="http://www.springframework.org/schema/beans | ||
06 | http://www.springframework.org/schema/beans/spring-beans-3.0.xsd"> | ||
07 | <bean id="car1" class="com.baobaotao.Car" | |
08 | p:brand="红旗CA72" |
09 | p:color="黑色" | |
10 | p:maxSpeed="200" /> |
11 | </beans> |
java调用代码:
1 | ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver(); |
2 | Resource res = resolver.getResource("classpath:com/coder/beanfactory/beans.xml"); |
3 | BeanFactory bf = new XmlBeanFactory(res); | |
4 | Car car = bf.getBean("car",Car.class); |
5 | System.out.println("car bean is ready for use!"); |
XmlBeanFactory通过Resource装载Spring配置信息并启动IoC容 器,然后就可以通过BeanFactory#getBean(beanName)方法从IoC容器中获取Bean了。通过BeanFactory启动 IoC容器时,并不会初始化配置文件中定义的Bean,初始化动作发生在第一个调用时。对于单实例(singleton)的Bean来 说,BeanFactory会缓存Bean实例,所以第二次使用getBean()获取Bean时将直接从IoC容器的缓存中获取Bean实例。
Spring在DefaultSingletonBeanRegistry类中提供了一个用于缓存单实例Bean的缓存器,它是一个用HashMap实现的缓存器,单实例的Bean以beanName为键保存在这个HashMap中。
ApplicationContext的主要实现类是ClassPathXmlApplicationContext和FileSystemXmlApplicationContext,前者默认从类路径加 载配置文件,后者默认从文件系统中装载配置文件,我们来了解一下ApplicationContext的类继承体系:
和BeanFactory初始化相似,ApplicationContext的初始化也很简单:
◆如果配置文件放置在类路径下,用户可以优先使用ClassPathXmlApplicationContext实现类:
1 | ApplicationContext ctx = new ClassPathXmlApplicationContext("com/coder/context/beans.xml"); |
对于ClassPathXmlApplicationContext来说,“com/coder/context/beans.xml”等同于“classpath:com/coder/context/beans.xml”。
◆如果配置文件放置在文件系统的路径下,则可以优先考虑使用FilySystemXmlApplicationContext实现类:
1 | ApplicationContext ctx = new FileSystemXmlApplicationContext("com/coder/context/beans.xml"); |
对于FileSystemXmlApplicationContext来说,“com/coder/context/beans.xml”等同于“file:com/coder/context/beans.xml”。
◆还可以指定一组配置文件,Spring会自动将多个配置文件在内存中"整合"成一个配置文件,如下所示:
1 | ApplicationContext ctx = new ClassPathXmlApplicationContext(new String[]{"conf/beans1.xml","conf/beans2.xml"}); |
在获取ApplicationContext实例后,就可以像BeanFactory一样调 用getBean(beanName)返回Bean了。ApplicationContext的初始化和BeanFactory有一个重大的区 别:BeanFactory在初始化容器时,并未实例化Bean,直到第一次访问某个Bean时才实例目标Bean;而ApplicationContext则在初始化应用上下文时就实例化所有单实例的Bean。因此ApplicationContext的初始化时间会比 BeanFactory稍长一些,不过稍后的调用则没有"第一次惩罚"的问题。
Spring 3.0支持基于类注解的配置方式,和基于XML文件配置方式的优势在于,类注解的配置方式可以很容易地让开发者控制Bean的初始化过程,比基于XML的 配置更加灵活。 Spring为基于注解类的配置提供了专门的ApplicationContext实现 类:AnnotationConfigApplicationContext。来看一个如何使用AnnotationConfigApplicationContext启动Spring容器的示例:
带注解的Java类提供的配置信息:
01 | //①表示是一个配置信息提供类 | |
02 | @Configuration public class Beans { |
03 | //②定义一个Bean | |
04 | @Bean(name = "car") |
05 | public Car buildCar() { | |
06 | Car car = new Car(); |
07 | car.setBrand("红旗CA72"); | |
08 | car.setMaxSpeed(200); |
09 | return car; | |
10 | } |
11 | } |
通过带@Configuration的配置类启动容器:
1 | //①通过一个带@Configuration的POJO装载Bean配置 | |
2 | ApplicationContext ctx = new AnnotationConfigApplicationContext(Beans.class); |
3 | Car car = ctx.getBean("car",Car.class); |
WebApplicationContext是专门为Web应用准备的,它允许从相对于Web根目录的路径中装载配置文件完成初始化工作。
从WebApplicationContext中可以获得ServletContext的引用,整个Web应用上下文对象将作为属性放置到ServletContext中,以便Web应用环境可以访问Spring应用上下文。 可以通过Spring提供的工具类WebApplicationContextUtils,获取WebApplicationContext实例:
1 | WebApplicationContext ctx = WebApplicationContextUtils.getRequiredWebApplicationContext(ServletContext sc); | |
2 | WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(ServletContext sc); | |
上面两个工具方式的区别是,前者在获取失败时抛出异常,后者返回null。
由于Web应用比一般的应用拥有更多的特性,因此WebApplicationContext扩展了ApplicationContext。WebApplicationContext定义了一个常量ROOT_WEB_APPLICATION_ CONTEXT_ATTRIBUTE,在上下文启动时,WebApplicationContext实例即以此为键放置在ServletContext的 属性列表中,因此我们可以直接通过以下语句从Web容器中获取WebApplicationContext:
1 | WebApplicationContext wac = (WebApplicationContext)servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE); |
WebApplicationContext初始化
WebApplicationContext的初始化方式和BeanFactory、ApplicationContext有所区别,因为WebApplicationContext需要ServletContext实例,也就是说它必须在拥有Web容器的前提下才能完成启动的工作。
Spring分别提供了用于启动WebApplicationContext的Servlet和Web容器监听器:
◆org.springframework.web.context.ContextLoaderServlet;
◆org.springframework.web.context.ContextLoaderListener。
两者的内部都实现了启动WebApplicationContext实例的逻辑,我们只要根据Web容器的具体情况选择两者之一,并在web.xml中完成配置就可以了:
通过Web容器监听器引导:
01 | … | |
02 | <!--①指定配置文件--> |
03 | <context-param> | |
04 | <param-name>contextConfigLocation</param-name> |
05 | <param-value>/WEB-INF/coder-dao.xml,/WEB-INF/coder-service.xml</param-value> | ||
06 | </context-param> | ||
07 | … | |
08 | <!--②声明Web容器监听器--> |
09 | <listener> | |
10 | <listener-class>org.springframework.web.context.ContextLoaderListener </listener-class> |
11 | </listener> |
通过自启动的Servlet引导:
01 | … | |
02 | <context-param> |
03 | <param-name>contextConfigLocation</param-name> | ||
04 | <param-value>/WEB-INF/coder-dao.xml,/WEB-INF/coder-service.xml</param-value> | ||
05 | </context-param> | |
06 | … |
07 | <!--①声明自动启动的Servlet --> | |
08 | <servlet> |
09 | <servlet-name>springContextLoaderServlet</servlet-name> | ||
10 | <servlet-class>org.springframework.web.context.ContextLoaderServlet</servlet-class> | ||
11 | <!--②启动顺序--> | |
12 | <load-on-startup>1</load-on-startup> |
13 | </servlet> |
如果使用标注@Configuration的Java类提供配置信息,则web.xml的配置需要按以下方式配置:
使用@Configuration的Java类提供配置信息的配置:
01 | <web-app> | |
02 | <!--通过指定context参数,让Spring使用AnnotationConfigWebApplicationContext而非XmlWebApplicationContext启动容器--> |
03 | <context-param> | |
04 | <param-name>contextClass</param-name> |
05 | <param-value>org.springframework.web.context.support.AnnotationConfigWebApplicationContext</param-value> | ||
06 | </context-param> | ||
07 | <!-- 指定标注了@Configuration的配置类,多个可以使用逗号或空格分隔--> | |
08 | <context-param> |
09 | <param-name>contextConfigLocation</param-name> | ||
10 | <param-value>com.coder.AppConfig1,com.coder.AppConfig1</param-value> | ||
11 | </context-param> | |
12 | <!-- ContextLoaderListener监听器将根据上面配置使用 AnnotationConfigWebApplicationContext根据contextConfigLocation指定的配置类启动Spring容器--> |
13 | <listener> | ||
14 | <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> | ||
15 | </listener> | |
16 | </web-app> |