Spring 中 Singleton 模式的线程安全
您目前处于:编程  2015年07月16日

Spring 中管理的 Bean 实例默认情况下是单例的(sigleton 类型),但 Spring 中的单例并不会影响应用的并发访问。

E.g. 从客户端传递到后台 controller - Service - Dao 这一个流程中,它们这些对象都是单例的,那么这些单例的对象在传递实体 bean 时不会出问题么?

由于实体 bean 不是单例的,并没有交给 Spring 来管理,每次都是手动的 New 出来的,所以即使那些处理数据的业务处理类是被多线程共享的,但是它们处理的数据并不是共享的,数据是每一个线程都有自己的一份,所以在数据这个方面是不会出现线程同步方面的问题的。

在 Controller 中引用的 Service,这些对象都是单例,但这些类里通常不会含有成员变量,所以不会出现线程同步问题了。

在一般情况下,只有无状态的 Bean 才可以在多线程环境下共享,在 Spring 中,绝大部分 Bean 都可以声明为 singleton 作用域,就是因为 Spring 对一些 Bean 中非线程安全状态采用 ThreadLocal 进行处理,让它们也成为线程安全的状态。

我们通过 JVM 的层面来分析 Spring 单例下的线程安全问题,下图:

我们知道 Struts2 不是单例模式的,每一个 Request 到达后台,Struts2 都会建立一个单独的线程,而 Spring 管理的 Bean 在后台应用启动时,则就会被初始化。

在 JVM 中,每一个 Thread 对应一个线程栈,每个方法对应线程栈中的一个栈帧。而 Spring 管理的 Bean 初始化被保存在 JVM 的 Heap 中。

当 Strut2 中的一个 Action 中引用 Spring 的 Bean 时,对应在每个线程栈中引用到的这个 Bean,都指向 Heap 中的同一个地址空间。假设这个 Bean 中有一个变量 temp,并有 setTemp 和  getTemp 方法,那么在多线程的情况下对 temp 赋值和取值,则会出现数据不一致的问题。


转载请并标注: “本文转载自 linkedkeeper.com ”