论坛首页 Java企业应用论坛

由Spring管理的Struts2的Action的单实例问题

浏览 19450 次
精华帖 (2) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (1)
作者 正文
   发表时间:2009-03-28   最后修改:2009-03-29

背景 :


1) Struts2 会对每一个请求,产生一个Action的实例来处理.

2) Spring的Ioc容器管理的bean默认是单实例的.


当Struts2与Spring整合后,由spring来管理Struts2的Action,会遇到什么问题 ?如何解决 ?

----------------------------------------------------------------


会遇到什么问题?


Struts2与Spring整合后, 由spring来管理Struts2的Action,   bean默认是单实例有情况下,会有如下问题:

1) Action是单例,其中的FieldError,actionerror中的错误信息 会累加, 即使再次输入了正确的信息,也过不了验证.

2) Struts2的Action是有状态的,他有自己的成员属性, 所以在多线程下,会有问题.


----------------------------------------------------------------

 

如何解决?


方案一: 就是不用单例, spring中bean的作用域设为prototype,每个请求对应一个实例.


方案二: spring中bean的作用域设为session ,每个session对应一个实例,解决了多线程问题.(如何设置作用域请看: 4 spring中bean的作用域 )

再写一个拦截器,  清空 FieldError与actionerror

 

源自网络

public class ClearFieldErrorInterceptor extends AbstractInterceptor {

@Override
public String intercept(ActionInvocation invocation) throws Exception {
ActionSupport actionSupport = (ActionSupport)invocation.getAction();
actionSupport.clearErrorsAndMessages();
String resultCode = invocation.invoke();
return resultCode;
} 
 

-------------------------------------------------------------------------------------


总结 :

 

方案一 , bean的作用域设为prototype,  担心性能不好, 但没实际测试过,不好说话,也只是担心而已.


方案二:  由于对方案一有担心, 所有才有了方案二, 不知比方案一性能 能高多少


有时间 ,再测试 一下.









   发表时间:2009-03-28  
...........................
Action 单例的话很带来很多问题的...很多东西都要自己处理

首先action就不应该带有状态了..
如你说的用拦截器去清理状态感觉不是很合适,一般用threadlocal去存那些状态.然后用拦截器去请这个threadlocal就可以了


还有...为啥用spring整合Struts2后action会变单例呢?
这好像是两码事吧...
0 请登录后投票
   发表时间:2009-03-28  
请问你用session和单例有什么区别没有?????


提示你一下:
如果我打开两个浏览器同时提交向一个action会出现什么???

0 请登录后投票
   发表时间:2009-03-28  
单例根本就不可行。。。主要是线程安全问题。
如果是单例,Action 中数据就是共享的。在多线程下,这种错误是致命的。
0 请登录后投票
   发表时间:2009-03-28  
hantsy 写道
单例根本就不可行。。。主要是线程安全问题。
如果是单例,Action 中数据就是共享的。在多线程下,这种错误是致命的。



Struts1 就是单例的..如果不可行.就不会这么多人用了...
只要处理的好还是可以用的....
问题就是要处理的好..线程相关的要自己处理了.
0 请登录后投票
   发表时间:2009-03-28  
zephyrleaves 写道
hantsy 写道
单例根本就不可行。。。主要是线程安全问题。
如果是单例,Action 中数据就是共享的。在多线程下,这种错误是致命的。



Struts1 就是单例的..如果不可行.就不会这么多人用了...
只要处理的好还是可以用的....
问题就是要处理的好..线程相关的要自己处理了.

s1跟s2在action的处理上是完全不一样的
s1如果如果设置scope是request,每次请求actionform是新的
s2则是用action取代了s1的actionform和action的功能,提交的数据和渲染到页面的数据都在action里,这是没法共享的,所以s2的action也就不能用单例
0 请登录后投票
   发表时间:2009-03-28  
这2天我也恰好考虑到LZ的问题。我倒是觉得struts2的action每次都create并不能满足所有的需求,我就是要在action里面放application或者session级别的共享数据如何?感谢spring,可以把我的action设置成为不同的作用域,让我的action在application和session中共享。
0 请登录后投票
   发表时间:2009-03-28  
hantsy 写道
单例根本就不可行。。。主要是线程安全问题。
如果是单例,Action 中数据就是共享的。在多线程下,这种错误是致命的。

无状态的对象单态没问题,如struts1的action,又如一般意义上的service.
另外一个方向来说,如果我的需求就是要让这个action数据全局共享,显然web框架该给我提供这个功能。
一如JSF的backed bean,Action应该可以设置作用域,决定是否可以共享。
0 请登录后投票
   发表时间:2009-03-28  
zephyrleaves 写道
hantsy 写道
单例根本就不可行。。。主要是线程安全问题。
如果是单例,Action 中数据就是共享的。在多线程下,这种错误是致命的。



Struts1 就是单例的..如果不可行.就不会这么多人用了...
只要处理的好还是可以用的....
问题就是要处理的好..线程相关的要自己处理了.


Struts1是有ActionForm
Struts2对应的是Action中的一个成员。

用单例不怕把成员实例覆盖么?
0 请登录后投票
   发表时间:2009-04-30  
为什么要在spring中配置action呢???这是robbin曾经说过的 哈哈
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics