Spring可以被看作是个一个大型工厂, 这个工厂的作用就是生产和管理Sping容器中的Bean。如果想要在项目中使用这个工厂,就需要开发者对Spring的配置文件进行配置。
Spring容器支持 XML 和 Properties 两种格式的配置文件,在实际开发中,最常使用的就是XML格式的配置方式。这种配置方式通过XML文件来注册并管理Bean之间的依赖关系。
在Spring中,XML配置文件的根元素
标签, 如下表所示。中包含了多个 子元素,每一个 子元素定义了一个Bean,并描述了该Bean如何被装配到Spring容器中。 元素中同样包含了多个属性以及子元素,
属性或子元素名称 描述 id 是一个Bean的唯一标识符,Spring容器对Bean的配置、管理都通过该 (id) 属性来完成。 name Spring容器同样可以通过此 (name) 属性对容器中的Bean进行配置和管理,name属性中可以为Bean指定多个名称,每个名称之间用逗号或分号隔开。 class 该属性指定了Bean的具体实现类,它必须是一个完整的类名,使用类的全限定名。
(如果Bean中没指定id和name,spring会将class值当做id来使用)scope 用来设定Bean实例的作用域,其属性值有: singleton(单例)、prototype(原型)、request、session、global Session、application 和websocket。
其默认值为singleton。constructor-arg 元素的子元素,可以使用此元素传入构造参数进行实例化。该元素的index属性指定构造参数的序号(从0开始),type属性指定构造参数的类型,参数值可以通过ref属性 或value属性直接指定,也可以通过ref或value子元素指定。 property 元素的子元素,用于调用Bean实例中的stter方法完成属性赋值,从而完成依赖注入。该元素的name属性指定Bean实例中的相应属性名,ref 属性或value属性用于指定参数值。 ref property、constructor-arg等元素的属性或子元素,可以用于指定Bean工厂中的某个Bean的实例。 value property、constructor-arg等元素的属性或子元素,可以用于直接指定一个常量值。 list 用于封装List或数组类型的依赖注入。
set 用于封装Set类型属性的依赖注入。 map 用于封装Map类型属性的依赖注入。 entry 在配置文件中,通常一个普通的Bean只需要定义id(或name)和class两个属性即可,例子如下:
上述例子中,分别使用配置文件中的id属性和name属性定义了两个bean,并用class元素指定其对应的实现类。(如果Bean中没指定id和name,spring会将class值当做id来使用)
在面向对象的程序中,想要使用某个对象,就需要先实例化这个对象。同样,在Spring中,要想使用容器中的Bean,也需要实例化Bean。
实例化Bean有三种方式,分别为①构造器实例化 (最常用)、② 静态工厂方式实例化 和 ③实例工厂方式实例化,其中最常用的是构造器实例化。
ps : 下面只具体讲解常用的 构造器实例化。
构造器实例化是指Spring容器通过Bean对应类中默认的无参构造方法来实例化Bean。
在com.instance.constructor包下创建Bean1.java 和 bean1.xml
Bean1.java
public class Bean1 { }bean1.xml
InstanceTest1.java
public class InstanceTest1 { public static void main(String[] args) { //创建管理Bean的ApplicationContext (其底层也是IOC容器在管理Bean) ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/instance/constructor/bean1.xml"); //获得容器职工存储的Bean Bean1 bean = (Bean1) applicationContext.getBean("bean1"); System.out.println(bean); } }
通过Spring容器创建一个Bean的实例时,不仅可以完成Bean的实例化,还可以为Bean指定特定的作用城。
Spring4.3中为 Bean的实例定义了7种作用域,如下表所示。
作用域名称 说明 singleton
(单例)使用singleton定义的Bean在Spring容器中将 只有一个实例,也就是说,无论有多少个Bean引用它,始终将指向同一个对象(spring容器创建一个Bean实例,共享使用,使用的都是同一个Bean)。这也是Spring容器默认的作用域。 prototype
(原型)每次通过Spring容器获取的prototype定义的Bean时,容器都将 创建一个新的Bean实例。 request 在一次HTTP请求中,容器会返回该Bean的同一个实例。对不同的HTTP请求则会产生一个新的Bean,而且该Bean仅在当前HTTP Request内有效。 session 在一次HTTP Session中,容器会返回该Bean的同一个实例。对不同的HTTP请求则会产生一个新的Bean,而且该Bean仅在当前HTTP Session内有效。 globalSession 在一个全局的HTTP Session中,容器会返回该Bean的同一个实例。 仅在使用portle上下文时有效。 application 为每个SenletContext对象创建一个实例。仅在Web相关的AplcatinConext中生效。 websocket 为每个websocket对象创建一个实例。仅在Web相关的AplcationContext中生效。
singleton是Spring容器默认的作用域 (创建一个Bean实例,然后共享/共同使用这个bean),当Bean的作用域为singleton时,Spring 都只会存在一个共享的Bean实例(一个Bean被共享使用),并且所有对Bean的请求,只要id 与 该Bean的id属性相匹配就会返回同一个Bean实例。
singleton 作用域对于无会话状态的Bean (如Dao组件、Service组件)来说,是最理想的选择。
在Spring配置文件中,Bean的作用域是通过
元素的scope属性来指定的 ,该属性可以设置为singleton、prototype、request、session、globalSession、application 和 websocket 七个值,分别表示表Bean的七种作用域。例子如下 :Scope.java
package com.scope; public class Scope { }1 applicationContext.xml
1 ScopeTest.java
package com.scope; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class ScopeTest { public static void main(String[] args) { //1.配置文件路径 String xmlPath = "com/scope/applicationContext.xml"; //2.加载配置文件 ApplicationContext applicationContext = new ClassPathXmlApplicationContext(xmlPath); //输出实例 System.out.println(applicationContext.getBean("scope")); //com.scope.Scope@737996a0 System.out.println(applicationContext.getBean("scope")); //com.scope.Scope@737996a0 /* 此处从spring容器中获得了两个bean,但id是一致的,且设置的作用域范围是: singleton, 所以这两个Bean都是同一个Bean的实例,两个Bean的地址信息都是一致的。 */ } }ScopeTest测试类运行结果如下:
从图可以看出 ,两次输出 的结果相同,这说明Spring容器只创建了一个Scope类的实例。需要注意的是,如果不设置 scope=“singleton”,其输出结果也是一个实例,因为Spring容器默认的作用域就是singleton。
对需要保持会话状态的Bean (如Struts2的Action 类) 应该使用prototype作用域。在使用prototype作用域时,Spring 容器会为每个对该Bean的请求都创建一个新的实例 (每次都是实例化一个新的Bean)。
要将Bean定义为prototype 作用域,只需在配置文件中将
元素的scope属性值设置为prototype 即可,其示例代码如下。 将ApplicationContext中的Bean的作用域范围改为: prototype 后,测试类运行结果如下 :
由运行结果可知,两次输出的Bean的实例并不相同,这说明在prototype作用域下,创建了两个不同的Scope实例。