Servlet是在多线程环境下的。即可能有多个请求发给一个servelt实例,每个请求是一个线程。
struts下的action也类似,同样在多线程环境下。可以参考struts user guide: http://struts.apache.org/struts-action/userGuide/building_controller.html 中的Action Class Design Guidelines一节: Write code for a multi-threaded environment - Our controller servlet creates only one instance of your Action class, and uses this one instance to service all requests. Thus, you need to write thread-safe Action classes. Follow the same guidelines you would use to write thread-safe Servlets.
译:为多线程环境编写代码。我们的controller servlet指挥创建你的Action 类的一个实例,用此实例来服务所有的请求。因此,你必须编写线程安全的Action类。遵循与写线程安全的servlet同样的方针。
1.什么是线程安全的代码
在多线程环境下能正确执行的代码就是线程安全的。
安全的意思是能正确执行,否则后果是程序执行错误,可能出现各种异常情况。
2.如何编写线程安全的代码
很多书籍里都详细讲解了如何这方面的问题,他们主要讲解的是如何同步线程对共享资源的使用的问题。主要是对synchronized关键字的各种用法,以及锁的概念。
Java1.5中也提供了如读写锁这类的工具类。这些都需要较高的技巧,而且相对难于调试。
但是,线程同步是不得以的方法,是比较复杂的,而且会带来性能的损失。等效的代码中,不需要同步在编写容易度和性能上会更好些。
我这里强调的是什么代码是始终为线程安全的、是不需要同步的。如下:
1)常量始终是线程安全的,因为只存在读操作。
2)对构造器的访问(new 操作)是线程安全的,因为每次都新建一个实例,不会访问共享的资源。
3)最重要的是:局部变量是线程安全的。因为每执行一个方法,都会在独立的空间创建局部变量,它不是共享的资源。局部变量包括方法的参数变量。
struts user guide里有:
Only Use Local Variables - The most important principle that aids in thread-safe coding is to use only local variables, not instance variables , in your Action class.
译:只使用用局部变量。--编写线程安全的代码最重要的原则就是,在Action类中只使用局部变量,不使用实例变量。
总结:
在Java的Web服务器环境下开发,要注意线程安全的问题。最简单的实现方式就是在Servlet和Struts Action里不要使用类变量、实例变量,但可以使用类常量和实例常量。如果有这些变量,可以将它们转换为方法的参数传入,以消除它们。
注意一个容易混淆的地方:被Servlet或Action调用的类中(如值对象、领域模型类)中是否可以安全的使用实例变量?如果你在每次方法调用时
新建一个对象,再调用它们的方法,则不存在同步问题---因为它们不是多个线程共享的资源,只有共享的资源才需要同步---而Servlet和Action的实例对于多个线程是共享的。
换句话说,Servlet和Action的实例会被多个线程同时调用,而过了这一层,如果在你自己的代码中没有另外启动线程,且每次调用后续业务对象时都是先新建一个实例再调用,则都是线程安全的。
=================================================
以上是我找到令我自己满意的答案!!!
2)对构造器的访问(new 操作)是线程安全的”
我想只有这一句话就够了!!
非常全面,不错。
_________________________________________________________
对构造器的访问(new 操作)是线程安全的
每次调用后续业务对象时都是先新建一个实例再调用,则都是线程安全的
这是正确的,这只是在Jsp+JavaBeans架构下适用。
通过new创建对象有很多设计问题,因此,实战中我们现在基本不通过new来获得对象。
既然不通过new获得对象,那么如何保证Struts线程问题?
原则就是:
一个请求一个线程,一个业务对象,一个资源,就象专用通道一样。
这里涉及对象多例、单例等复杂问题。
-------------------------------------------------------------------------------
当用spring向struts里注入业务对象时,通常都是把这个业务对象作为成员变量注入
但是struts的文档上有这么一句话:
Actions must be programmed in a thread safe manner(方式), because the controller will share the same instance for multiple(多个) simultaneous(同时发生的) requests.Instance and static variables must not be used to store information related to the state of a particular request(译:由于在同时发生的多个请求中,controller会共享同一个实例(instance ),因此 Actions 必须在以一个线程安全的方式运行。)
线程安全的做法:
1。可以用同步
2。在spring的配置文件中将bean的singleton设为false,如<beanid=""class=""singleton="false">
这样在容器中每次产生的bean都是一个新的实例。