`

线程安全,有状态,无状态的对象

阅读更多

进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单位.
线程是进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.
另外,进程在执行过程中拥有独立的内存单元,而多个线程共享内存,从而极大地提高了程序的运行效率。
进程有独立的地址空间


进程有单独的地址空间,包括文本区域(text region)、数据区域(data region)和堆栈(stack region)。
文本区域存储处理器执行的代码;
数据区域存储变量和进程执行期间使用的动态分配的内存;
堆栈区域存储着活动过程调用的指令和本地变量。


线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间
一个线程包含以下内容。

•一个指向当前被执行指令的指令指针;
•一个栈;
•一个寄存器值的集合,定义了一部分描述正在执行线程的处理器状态的值;
•一个私有的数据区。

线程安全 


如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。   或者说:一个类或者程序所提供的接口对于线程来说是原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题。   线程安全问题都是由全局变量及静态变量引起的。  
若每个线程中对全局变量、静态变量只有读操作,而无写操作,一般来说,这个全局变量是线程安全的;若有多个线程同时执行写操作,一般都需要考虑线程同步,否则就可能影响线程安全。

 

关于线程安全:

 

1) 常量始终是线程安全的,因为只存在读操作。 

 

2)每次调用方法前都新建一个实例是线程安全的,因为不会访问共享的资源。

 

3)局部变量是线程安全的。因为每执行一个方法,都会在独立的空间创建局部变量,它不是共享的资源。局部变量包括方法的参数变量和方法内变量。 


有状态和无状态对象

有状态就是有数据存储功能。有状态对象(Stateful Bean),就是有实例变量的对象  ,可以保存数据,是非线程安全的。在不同方法调用间不保留任何状态。

无状态就是一次操作,不能保存数据。无状态对象(Stateless Bean),就是没有实例变量的对象  .不能保存数据,是不变类,是线程安全的。

有状态对象:

Java代码  收藏代码
  1.  public   class  StatefulBean {     
  2.      
  3.      public   int  state;     
  4.    
  5.      public   int  getState() {     
  6.          return  state;     
  7.     }     
  8.     
  9.      public   void  setState( int  state) {     
  10.          this .state = state;     
  11.     }     
  12.     
  13.      public  User getUser() {     
  14.          return  user;     
  15.     }     
  16.     
  17.      public   void  setUser(User user) {     
  18.          this .user = user;     
  19.    }     
  20. }     

 无状态对象

Java代码  收藏代码
  1. public   class  StatefulBean {     
  2.     
  3.    // 虽然有userDao 属性,但userDao 是没有状态信息的,是Stateless Bean。  
  4.     public   UserDao  userDao  ;     
  5.   
  6.     public   int  getState() {     
  7.         return  state;     
  8.    }     
  9.    
  10.     public   void  setState( int  state) {     
  11.         this .state = state;     
  12.    }     
  13.    
  14.     public  User getUser() {     
  15.         return  user;     
  16.    }     
  17.    
  18.     public   void  setUser(User user) {     
  19.         this .user = user;     
  20.   }     
  21.      

 单例类可以是有状态的(stateful),也可以是无状态的。无状态的单例模式,是线程安全的。有状态的单例模式,是非线程安全的。

 

 

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作用域。

 

 

 

Struts中的线程安全

 

1.Struts1也是基于单例模式实现,也就是只有一个Action实例供多线程使用。默认的模式是前台页面数据通过actionForm传入,在 action中的excute方法接收,这样action是无状态的,所以一般情况下Strunts1是线程安全的。如果Action中用了实例变量,那 么就变成有状态了,同样是非线程安全的。

 

 

2.Struts2默认的实现是Prototype模式。也就是每个请求都新生成一个Action实例,所以不存在线程安全问题。需要注意的是,如果由Spring管理action的生命周期, scope要配成prototype作用域。

 

总结:   
Stateless无状态用单例Singleton模式,Stateful有状态就用原型Prototype模式。  
Stateful 有状态是多线程编码的天敌,所以在开发中尽量用Stateless无状态,无状态是不变(immutable)模式的应用,有很多优点:不用管线程和同步的问题  ,如果值是不可变的,程序不用担心多个线程改变共享状态,所以可以避免线程竞争的bugs. 因为没有竞争,就不用用locks等机制,所以无状态的不变机制,也可以避免产生死锁现象。

原文出处:

http://blog.csdn.net/showwair/article/details/7672443

分享到:
评论

相关推荐

    构建线程安全应用程序

    当对一个复杂对象进行某种操作时,从操作开始到操作结束,被操作的对象...如果其他线程企图访问一个处于不可用状态的对象,该对象将不能正确响应从而产生无法预料的结果,如何避免这种情况发生是线程安全性的核心问题

    当析构函数遇到多线程── C++ 中线程安全的对象回调

    编写线程安全的类不是难事,用同步原语保护内部状态即可。但是对象的生与死不能由对象自身拥有的互斥器来保护。如何保证即将析构对象 x 的时候,不会有另一个线程正在调用 x 的成员函数?或者说,如何保证在执行 x ...

    设计滚动字演示线程状态及改变方法

    设计滚动字演示线程状态及改变方法

    java多线程安全性基础介绍.pptx

    多个线程操作共享对象导致的状态不一致问题 原因 共享资源的竞态条件问题 问题 竞态条件 指令重排序 工作内存中主内存同步延迟 解决 要有安全策略文档或注释 不共享 线程封闭 仅在单线程内访问数据 栈...

    当析构函数遇到多线程── C++ 中线程安全的对象回调 PDF

    编写线程安全的类不是难事,用同步原语保护内部状态即可。但是对象的生与死不能由对象自身拥有的互斥器来保护。如何保证即将析构对象 x 的时候,不会有另一个线程正在调用 x 的成员函数?或者说,如何保证在执行 x ...

    python 线程的五个状态

    无论是通过 Thread 类直接实例化对象创建线程,还是通过继承自 Thread 类的子类实例化创建线程,新创建的线程在调用 start() 方法之前,不会得到执行,此阶段的线程就处于新建状态。 线程的就绪状态 当位于新建状态...

    Java线程的6种状态及切换(透彻讲解).docx

    1. 初始(NEW) :新创建了一个线程对象,但还没有调用start()方法。 2. 运行(RUNNABLE) :Java线程中将就绪(ready)和运行中(running)两种状态笼统的称为“运行”。 线程对象创建后,其他线程(比如main线程)...

    C++ 创建线程互斥对象

    C++ 创建线程互斥对象的实例源码下载,声明线程函数,创建线程,程序睡眠,释放互斥对象,设置事件对象为无信号状态,生成控制台程序,仅供参考。

    Java并发编程规则:不可变对象永远是线程安全的.docx

    创建后状态不能被修改的对象叫作不可变对象。不可变对象天生就是线程安全的。它们的常量(变量)是在构造函数中创建的,既然它们的状态无法被修改,那么这些常量永远不会被改变——不可变对象永远是线程安全的。

    dynamic-pool:无锁,线程安全,动态大小的对象池

    无锁,线程安全,动态大小的对象池。 该池以初始容量开始,如果没有可用空间,它将根据请求继续创建新对象。 合并后的对象将在销毁后返回到池中(有一项额外规定,可以选择“重置”对象的状态以供重新使用)。 ...

    线程编程方面若标志指出线程应该挂起,便用wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。

    它会解除由线程获取的所有锁定,而且如果对象处于一种不连贯状态,那么其他线程能在那种状态下检查和修改它们。结果很难检查出真正的问题所在。suspend()方法容易发生死锁。调用suspend()的时候,目标线程会停下来,...

    《当析构函数遇到多线程》PPT

    简介:编写线程安全的类不是难事,用同步原语保护内部状态即可。但是对象的生与死不能由对象自身拥有的互斥器来保护。如何保证即将析构对象 x 的时候,不会有另一个线程正在调用 x 的成员函数?或者说,如何保证在...

    C 创建线程互斥对象的实例源代码.rar

    C 创建线程互斥对象的实例源码下载,声明线程函数,创建线程,程序睡眠,释放互斥对象,设置事件对象为无信号状态,生成控制台程序,仅供参考。

    Java并发理论,如何理解线程安全.docx

    当多个线程访问同一个对象时,如果不用考虑这些线程在运行环境下的调度和交替运行,也不需要进行额外的同步,或者在调用方进行任何其他的协调操作,调用这个对象的行为都可以获取正确的结果,那么这个对象时线程的 ...

    线程运行状态

    CPU执行调度到某个线程对象后,该线程对象开始执行run方法或call方法,但是不会一直占用资源,而是在一段时间之后(一个时间片的时间之后),该线程就需要暂停执行,如果这时线程还没执行完,那么就由阻塞状态切换到...

    Java 线程状态、线程池

    java 线程状态、线程池 1. java 的线程状态 状态 发生条件 NEW 线程刚刚被创建,没有启动,没有调用start方法 RUNNABLE(可运行) 线程已经在JVM中运行,但是是否运行不确定,看当前线程是否由CPU执行权 ...

    JAVA并发编程实践-线程安全-学习笔记

    线程安全就是对共享的、可变的状态进行管理,对象的状态就是它的数据,换句话说就是在不可控制的并发访问中保护数据。

    PoolSharp:PoolSharp是一个简单,轻量,线程安全的对象池

    PoolSharp是一个简单,轻量,线程安全的对象池。 它还支持一次性类型的合并,管理合并对象的生存时间并在可能的情况下进行早期处理。 池实现实现了一个简单的通用接口,因此可以对其进行模拟或替换。支持平台现在; ...

    c++调用PYthon线程状态

    为了支持多线程Python程序,当前线程在安全访问Python对象之前,必须持有全局锁(GIL,Global Interpreter Lock,全局解释器锁)。如果没有锁,多线程中甚至最简单的操作都会发生问题。例如,两个线程同时增加一个...

    Linux 线程间同步机制

    (1)在访问该资源前,首先申请该互斥锁,如果该互斥处于开锁状态,则申请到该锁对象,并立即占有该锁(使该锁处于锁定状态),以防止其它线程访问该资源;如果该互斥锁处于锁定状态,默认阻塞等待; (2)只有锁定...

Global site tag (gtag.js) - Google Analytics