论坛首页 Java企业应用论坛

在Webwork中使用ww:action实现页面控制器风格

浏览 10739 次
该帖已经被评为精华帖
作者 正文
   发表时间:2006-03-10  
标题:在Webwork中使用ww:action实现页面控制器风格Action复用

Webwork是标准的请求风格Web MVC,类似的有Struts、Spring MVC。这这种风格的MVC中都使用了前端控制器模式(企业架构模式),也就是说一个URL会被解析然后派发到对应的Action解析,而View调用的是Action处理后的Form对象或者Command对象(Rod的Without EJB)。
上面引用了一些经典的言论,而它们和“页面控制器风格Action复用”有什么关系呢?
嘿嘿,页面控制器在这里指到了View的Page里面依然可以调用控制器(在Webwork中就是Action)。如果是以前肯定有人会说,如果用JSP我随时可以在Page里面写scriptlet,什么东西都能调用……
可是,现在不一样了。我们希望让View单纯一点,不要有杂乱无章的逻辑参与其中。
可是,如果要是如上面所说那么Action在它的一个请求生命周期就要做所有的事……这样一是使Action逻辑变得复杂,二是会变得难以复用。
所以,说到这里就引出了这个主题:
1、在页面调用控制器
2、复用控制器逻辑
在Webwork中我们可以使用<ww:action/>标签实现这个目的。
——————————————————————————————————————
不要着急,我们先介绍一下ww:action的两种主要用法,而说用法之前先说语法:
语法:
<ww:action/>有5个属性:
1、id:给Action返回的ValueStack命名,如果不写则默认为调用的Action的名字。(详细使用参照后面)
2、name:调用的Action的name。
3、namespace:调用的Action的namespace。
4、executeResult:true或false,是否渲染Action的View。这个决定了ww:action的用法。(后面会做说明)
5、ignoreContextParams:Boolean值,request参数是否在Action被调用时所包括。

好了,语法很简单,我们说ww:action的两种主要用法。
1、代替<jsp:include>:
include有两种方式@ include和jsp:include,它们一个是编译前一个是运行时include。webwork是不能使用<jsp:include>的。
但其实ww有ww:include标签,但是根据Webwork in Action中的推荐,ww:include标签适合调用一般servlet,而对于action则推荐使用强大的ww:action。所以我们这里就略过ww:include。
说强大是什么意思呢?ww:action充当这个角色时,可以选择是否将valueStack的东西复制过来。
当ww:action代替jsp:include的时候我们需要executeResult="true",这个时候调用的action返回的view会被include到调用的位置。<ww:action><param name="xxx" value="yyy"/></ww:action>则可以给action传递参数。其它的用法就与jsp:include或者ww:include用法差别不大了。

2、页面控制器风格Action复用:
我们经常遇到这样的场景,比如用户注册的时候需要选择单位列表。那么我们reg.action运行之前就需要先把单位列表unitsList取出来。而它们本身与User注册逻辑上没什么关系。
所以有的人把这个取出unitsList单独写在prepare()方法里面,然后用prepare Inteceptor……或者把读取unitsList的逻辑写在execute方法里面。
但是这显然难以复用!
其实如果有单位unit这样的domain,我们可能就有对应的CRUD的Action。其中可能就有UnitsListAction这样的Action。
我们完全可以在用户注册的时候就复用这个Action,而不是把同样的逻辑写到用户注册的Action里面。这就是页面控制器风格要解决的问题。
说那么多大帽子其实没有意义,我们看看怎么实现:
UnitsListAction片断(我们要复用它):

UnitService unitService = null;//注入,商业逻辑
List<Unit> unitsList = null;//设置对应getter、setter

Public String execute(); {
 unitsList = unitDao.listAll();;
 return SUCCESS;
}


UserRegAction假设在注册前只是doDefault()直接返回SUCCESS,只有在Post数据时在调用execute(),我们就不写空的代码了。或者不通过任何Action调用注册页面,直接调用注册的jsp文件直接访问也可以。

到了UserRegAction显示的View,我这里是Jsp片断:
<ww:action id="listUnits" executeResult="false" namespace="/" name="unitsListAction" />
<ww:select name="unitId" list="#attr.listUnits.unitsList" listKey="id" listValue="name" required="true"/>

注意,executeResult="false",也就是说我们不渲染unitsListAction返回的view,只用它的值。
而访问它的值的时候要使用#attr.listUnits.unitsList这样的引用,因为这时unitsListAction返回的VlueStack不是页面的ognl的rootStack,我们需要访问#attr这个Stack,这部分可以参考一下Webwork的wiki。
上面我给unitsListAction规定了一个id,这样调用比较灵活,你可以多次调用同一个Action并且将值放在#attr下的不同地方。
我们引用unitsListAction返回的unitsList这个list的时候需要用#attr加上我们给unitsListAction设定的id(如果不指定id,则默认unitsListAction)再加上你要访问的变量名访问。
其实很简单,而这种方式就是开始说的页面控制器风格的action复用。虽然和真正的叶面控制器的Tapestry和JSF相差甚远,但是也算有点那个意思了。
扩展点想,如果我们在Action实现了一个counter,也可以通过这种方式调用,连返回的值都可以忽略,呵呵,这种逻辑复用还是挺有用的。

抛砖引玉,我这个人说话啰嗦,见谅,就到这里。
   发表时间:2006-03-21  
我原来以为有个这个东西,就可以不用把得到的list放到Session中用来出错返回这个页面时能够正常显示。但是试了一下,好像在输入检查没通过返回来时这个action不会执行。这样页面还是不能正常显示,下拉列表中的值没有了。
0 请登录后投票
   发表时间:2006-05-29  
zjumty 写道
我原来以为有个这个东西,就可以不用把得到的list放到Session中用来出错返回这个页面时能够正常显示。但是试了一下,好像在输入检查没通过返回来时这个action不会执行。这样页面还是不能正常显示,下拉列表中的值没有了。

同问这个, 有没有比较好的方法.
0 请登录后投票
   发表时间:2006-05-29  
按照文档和自己以前做过的实验,需要在被调用的Action里显式语句将值放到valueStatck的Request里。

public class ActionTagAction extends ActionSupport {

public String execute() throws Exception {
return "done";
}

public String doDefault() throws Exception {
ServletActionContext.getRequest().setAttribute("stringByAction", "This is a String put in by the action's doDefault()");
return "done";
}
}
webwork的文档:
http://www.opensymphony.com/webwork/wikidocs/action.html
0 请登录后投票
   发表时间:2006-06-05  
zjumty 写道
我原来以为有个这个东西,就可以不用把得到的list放到Session中用来出错返回这个页面时能够正常显示。但是试了一下,好像在输入检查没通过返回来时这个action不会执行。这样页面还是不能正常显示,下拉列表中的值没有了。


奇怪呀,我一直是这样用的呀,没有问题呀。就是单独使用jsp访问也工作正常呀。在使用dispather和redirect两种result的时候和单独访问jsp的生命周期非常类似呀。为了出错返回使用session,我可不喜欢。
0 请登录后投票
   发表时间:2006-06-06  
Why not use Preparable?
0 请登录后投票
   发表时间:2006-06-06  
dwangel 写道
Why not use Preparable?


这个可以自己写个interceptor 在 DefaultWorkflowInterceptor 前 ValidationInterceptor 后 执行, 负责检查当前的action是否有错误,如果有错误就尝试调用 methodNameOnError() 方法, 一样可以增加一个 PreResultListener 在action执行结束后 判断是否有错误, 没有就执行methodNameOnSuccess()方法.  如果单纯是用preparable 可能有效率问题, 应为prepare不关心我action执行是否有错误都执行.
0 请登录后投票
   发表时间:2006-06-06  
嗯,同意,最好继承MethodFilterInterceptor
这样可以在配置中设置忽略的方法。
0 请登录后投票
论坛首页 Java企业应用版

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