封封窝

还没有想好

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::
[转自Javaeye]

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片断(我们要复用它):

java代码: 

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片断:
java代码: 

<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,也可以通过这种方式调用,连返回的值都可以忽略,呵呵,这种逻辑复用还是挺有用的。
posted on 2006-05-30 12:55  封封  阅读(732)  评论(0编辑  收藏  举报