一、基本概念
1、有状态就是有数据存储功能。有状态对象(Stateful Bean),就是有实例变量的对象,可以保存数据,是非线程安全的。在不同方法调用间不保留任何状态。
2、无状态就是一次操作,不能保存数据。无状态对象(Stateless Bean),就是没有实例变量的对象.不能保存数据,是不变类,是线程安全的。
二、看代码加深印象
/** * 有状态bean,有state,user等属性,并且user有存偖功能,是可变的。 * * @author Peter Wei * */publicclassStatefulBean { publicintstate; // 由于多线程环境下,user是引用对象,是非线程安全的 publicUseruser; publicintgetState() { returnstate; } publicvoidsetState(intstate) { this.state=state; } publicUsergetUser() { returnuser; } publicvoidsetUser(Useruser) { this.user=user; } } /** * 无状态bean,不能存偖数据。因为没有任何属性,所以是不可变的。只有一系统的方法操作。 * * @author Peter Wei * */publicclassStatelessBeanService { // 虽然有billDao属性,但billDao是没有状态信息的,是Stateless Bean. BillDaobillDao; publicBillDaogetBillDao() { returnbillDao; } publicvoidsetBillDao(BillDaobillDao) { this.billDao=billDao; } publicList<User>findUser(StringId) { returnnull; } }
三、Spring中的有状态(Stateful)和无状态(Stateless)
1、通过上面的分析,相信大家已经对有状态和无状态有了一定的理解。无状态的Bean适合用不变模式,技术就是单例模式,这样可以共享实例,提高性能。有状态的Bean,多线程环境下不安全,那么适合用Prototype原型模式。Prototype: 每次对bean的请求都会创建一个新的bean实例。
2、默认情况下,从Spring bean工厂所取得的实例为singleton(scope属性为singleton),容器只存在一个共享的bean实例。
3、理解了两者的关系,那么scope选择的原则就很容易了:有状态的bean都使用prototype作用域,而对无状态的bean则应该使用singleton作用域。
4、如Service层、Dao层用默认singleton就行,虽然Service类也有dao这样的属性,但dao这些类都是没有状态信息的,也就是相当于不变(immutable)类,所以不影响。Struts2中的Action因为会有User、BizEntity这样的实例对象,是有状态信息的,在多线程环境下是不安全的,所以Struts2默认的实现是Prototype模式。在Spring中,Struts2的Action中,scope要配成prototype作用域。
四、Servlet是单例模式
1、Servlet体系结构是建立在Java多线程机制之上的,它的生命周期是由Web 容器负责的。一个Servlet类在Application中只有一个实例存在,也就是有多个线程在使用这个实例。这是单例模式的应用。无状态的单例是线程安全的,但我们如果在Servlet里用了实例变量,那么就变成有状态了,是非线程安全的。如下面的用法就是不安全的,因为user,out都是有状态信息的。
/** * 非线程安全的Servlet。 * @author Peter Wei * */publicclassUnSafeServletHttpServlet{ Useruser; PrintWriterout; publicvoiddoGet (HttpServletRequestrequest, HttpServletResponseresponse)throwsServletException, IOException{ //do something... } }
Out、Request、Response、Session、Config、Page、PageContext 是线程安全的,Application 在整个系统内被使用,所以不是线程安全的。