通过之前两篇文章的总结,spring已经完成了对xml文件的解析以及对beanDefinition的定义。在进行解析标签的时候,bean标签是最为重要的,其中bean标签还涉及到很多常用的属性,本文主要对bean标签scope属性进行总结。
简介
scope用于标注bean的作用域,可以将其设置为singleton ,prototype,request
、session
、global session。下边对这几个属性值进行一一的分析,其中可能还会涉及到非spring内容。
-
singleton
将scope设置为singleton,表示在整个ioc容器中该bean是一个单例,也就是说在获取bean的时候都会返回同一个bean对象("=="为true)。但是要注意的是“在同一个ioc容器中”,何为一个ioc容器?
ApplicationContext context = new ClassPathXmlApplicationContext("spring-bean.xml");
context 即为一个ioc容器。因此从这里可以看出spring的单例和我们自己写的单例其实在有效范围方面是有所区别的。
public class Singleton{
private static Singleton single;
public static Singleton getInstance(){
if (single == null){
synchronized(this){
if (single == null)
single = new Singleton();
}
}
return single;
}
}
以上代码为很多人经常实现的所谓的线程安全的单例模式,为什么说是所谓的线程安全,后边再说。我们所实现的单例通常情况下的在整个项目中的单例模式,作用范围是整个项目,而不是和spring那样以ioc容器为范围。
上边代码实现是比较常用的double-check的懒汉式单例模式,但是其实它并不是线程安全的。主要问题就出现在:
single = new Singleton();
在创建对象的时候其实是分为三个非原子性操作的:
1 在堆空间分配内存
2 初始化内存
3 将引用指向该内存
由于编译器指令重排,可能执行顺序不是 1-2-3,而成了 1-3-2这样在多线程情况下就会出现问题。在并发访问时有可能会返回一个为初始化内存的引用(null)。下边代码为真正线程安全的单例模式。
public class Singleton{
private static volatile Singleton single;
public static Singleton getInstance(){
if (single == null){
synchronized(this){
if (single == null)
single = new Singleton();
}
}
return single;
}
}
-
prototype
与singleton相反,该值代表非单例,也就是说每次从ioc获取bean时都是一个新的bean实例。如果将scope属性设置为prototype则spring 不会管理bean的整个生命周期,获取时容器将初始化好的bean返回至客户端,但是销毁操作需要客户端执行,容器不会进行销毁操作,也就是说不会调用destroy方法,因此看起来和new 一个bean在某种意义上有点类似。
-
request
、session
、global session
这几种不做详细说明,不常用,也不复杂。