1.2. 容器概述
org.springframework.context.ApplicationContext
接口代表 Spring IoC 容器,并负责实例化,配置和组装 Bean。容器通过读取配置元数据来获取有关要实例化,配置和组装哪些对象的指令。配置元数据以 XML,Java 注解或 Java 代码表示。它使您能够表达组成应用程序的对象以及这些对象之间丰富的相互依赖关系。
Spring 提供了ApplicationContext
接口的几种实现。在独立应用程序中,通常创建ClassPathXmlApplicationContext或FileSystemXmlApplicationContext的实例。虽然XML一直是定义配置元数据的传统格式,但您可以通过提供少量XML配置来指示容器使用Java注释或代码作为元数据格式,以声明方式支持这些额外的元数据格式。
在大多数应用场景中,实例化 Spring IoC 容器的一个或多个实例不需要显式的用户代码。例如,在 Web 应用程序场景中,web.xml
文件中简单的8行左右的关于描述 web 的样板通常就足够了(请参阅Convenient ApplicationContext Instantiation for Web Applications。如果使用Spring Tools for Eclipse(基于 Eclipse 的开发环境),那么仅需单击几次鼠标或击键即可轻松创建此样板配置。
下图显示了 Spring 的如何工作的高级视图。您的应用程序类与配置元数据结合在一起,这样,在创建和初始化ApplicationContext
之后,您将具有完全配置和可执行的系统或应用程序。
Figure 1. The Spring IoC container
1.2.1 配置元数据
如上图所示,Spring IoC 容器使用一种配置元数据的格式。此配置元数据表示您作为应用程序开发人员如何告诉 Spring 容器实例化,配置和组装应用程序中的对象。
传统方法是使用简单直观的 XML 格式配置元数据,这是本章大部分内容以此方法传达 Spring IoC 容器的关键概念和功能。
Note
基于 XML 的元数据不是配置元数据的唯一允许格式。 Spring IoC 容器本身与实际写入此配置元数据的格式是完全脱钩的。如今,许多开发人员为自己的 Spring 应用程序选择Java-based configuration。
有关在 Spring 容器中使用其他形式的元数据的信息,请参见:
Annotation-based configuration:Spring 2.5 引入了对基于注解的配置元数据的支持。
Java-based configuration:从 Spring 3.0 开始,Spring JavaConfig 项目提供的许多功能成为了core Spring Framework 的一部分。因此,您可以使用 Java 而不是 XML 文件来定义应用程序类外部的 bean。要使用这些新功能,请参见@Configuration,@Bean,@Import和@DependsOn注解。
Spring 配置包含容器管理的至少一个bean定义,通常是多个bean定义。基于 XML 的配置元数据将这些 bean 配置为
注解的方法。
这些 bean 定义对应于组成应用程序的实际对象。通常,您定义服务层对象,数据访问对象(DAO),展示对象(例如 控制层实例),基础设施对象(例如 Hibernate SessionFactories
,JMS Queues
等)。通常,不会在容器中配置细粒度的域对象,因为 DAO 和业务逻辑通常负责创建和加载域对象。但是,您可以使用 Spring 与 AspectJ 的集成来配置在 IoC 容器的控制范围之外创建的对象。参见使用 Spring 与 AspectJ 来依赖注入对象。
以下示例显示了基于 XML 的配置元数据的基本结构:
- id 属性是标识单个 bean 定义的字符串。
- class 属性使用完整的类名并定义 bean 的类型。
id
属性的值涉及到有协作关系的其他对象。在此XML示例中未显示有协作关系的其他对象 。有关更多信息,请参见Dependencies。
1.2.2. 实例化容器
提供给ApplicationContext构造函数的 location path 或路径是resource strings,使容器从自各种外部资源加载配置元数据,例如本地文件系统,CLASSPATH等。
ApplicationContext context = new ClassPathXmlApplicationContext("services.xml", "daos.xml");
Note
了解了 Spring 的 IoC 容器之后,您可能想了解更多有关 Spring 的Resource
抽象(如Resources中所述),该抽象为从 URI 语法中定义的 locations 读取 InputStream 提供了一种方便的机制。特别是,Resource
路径用于构建应用程序上下文,如Application Contexts and Resource Paths中所述。
以下示例显示了服务层对象(services.xml)配置文件:
以下示例显示了数据访问对象daos.xml文件:
在前面的示例中,服务层由PetStoreServiceImpl
类和两个类型为JpaAccountDao
和JpaItemDao
的数据访问对象组成(基于 JPA 对象关系映射标准)。 property name
元素表示 JavaBean 属性 的名称,而ref
元素表示另一个 bean definition 的名称。 id
和ref
元素之间的这种联系表达了协作对象之间的依赖性关系。有关配置对象的依赖项的详细信息,请参见Dependencies。
构成基于 XML 的配置元数据
使 bean 定义跨越多个 XML 文件是很有用的。通常,每个单独的 XML 配置文件都代表架构中的逻辑层或模块。
您可以使用 application context 的构造函数从所有这些 XML 片段中加载 bean definition。该构造函数具有多个Resource
位置,如previous section所示。或者,使用一个或多个
元素从另一个文件中加载 bean 定义。以下示例显示了如何执行此操作:
在前面的示例中,从三个文件services.xml,messageSource.xml和themeSource.xml加载了外部 bean 定义。所有位置路径都相对于进行导入的定义文件,因此services.xml必须与进行导入的文件位于同一目录或 Classpath 位置,而messageSource.xml和themeSource.xml必须位于导入文件位置下方的resources位置。如您所见,斜杠被忽略。但是,鉴于这些路径是相对的,最好不要使用任何斜线。根据 Spring Schema,导入的文件的内容(包括顶级
Note
可以但不建议使用相对的“ ../”路径引用父目录中的文件。这样做会创建对当前应用程序外部文件的依赖关系。特别是,不建议对classpath: URL(例如classpath:../services.xml)使用此引用,在 URL 中,运行时解析过程选择“最近”Classpath 根,然后查看其父目录。Classpath 配置的更改可能导致选择其他错误的目录。
您始终可以使用标准资源位置而不是相对路径:例如file:C:/config/services.xml或classpath:/config/services.xml。但是,请注意,您正在将应用程序的配置耦合到特定的绝对位置。通常,最好为这样的绝对位置保留一个间接寻址,例如通过在运行时针对 JVM 系统属性解析的“ ${…}”占位符。
名称空间本身提供了导入指令功能。 Spring 提供的一系列 XML 名称空间(例如context和util名称空间)中提供了超出普通 bean 定义的其他配置功能。
The Groovy Bean Definition DSL
略
1.2.3. 使用容器
ApplicationContext是高级工厂的接口,能够维护不同bean及其依赖项的注册表。通过使用方法 T getBean(String name, Class
ApplicationContext允许您读取 bean 定义并访问它们,如以下示例所示:
// create and configure beans
ApplicationContext context = new ClassPathXmlApplicationContext("services.xml", "daos.xml");
// retrieve configured instance
PetStoreService service = context.getBean("petStore", PetStoreService.class);
// use configured instance
List userList = service.getUsernameList();
对于Groovy配置,引导看起来非常相似。它有一个不同的上下文实现类,它可以感知Groovy(同时也理解XML Bean定义)。下面的示例显示Groovy配置:
ApplicationContext context = new GenericGroovyApplicationContext("services.groovy", "daos.groovy");
最灵活的变体是GenericApplicationContext与 读取器 结合使用,例如,对于 XML 文件,是XmlBeanDefinitionReader,如以下示例所示:
GenericApplicationContext context = new GenericApplicationContext();
new XmlBeanDefinitionReader(context).loadBeanDefinitions("services.xml", "daos.xml");
context.refresh();
您还可以将GroovyBeanDefinitionReader用于 Groovy 文件,如以下示例所示:
GenericApplicationContext context = new GenericApplicationContext();
new GroovyBeanDefinitionReader(context).loadBeanDefinitions("services.groovy", "daos.groovy");
context.refresh();
您可以在相同的ApplicationContext上混合并匹配此类阅读器委托,从不同的配置源读取 Bean 定义。
然后,您可以使用getBean来检索 bean 的实例。 ApplicationContext接口还有其他几种检索 bean 的方法,但是理想情况下,您的应用程序代码永远不要使用它们。实际上,您的应用程序代码应该根本不调用getBean()方法,因此完全不依赖于 Spring API。例如,Spring 与 Web 框架的集成为各种 Web 框架组件(例如controllers 和 JSF-managed beans)提供了依赖注入,使您可以通过元数据(例如自动装配)声明对特定 Bean 的依赖。