论坛首页 Java企业应用论坛

Pragmatic Struts

浏览 28372 次
该帖已经被评为精华帖
作者 正文
   发表时间:2006-11-17  

 本文来自SpringSide WIki,请留意Wiki上的最新版本

  初学struts, 抛块砖头出来,请各位常年与Struts作战的兄弟点评。

   Struts 1.2 其实已经大不同,只要大家有简约的态度,即使我们日常的Struts,同样可以写得非常精简,不一定要如传统观念认为的那么繁重,从而节省每天编码的时间。

    Struts与Spring结合的基本知识请阅读Struts使用要点

1.DispatchAction

     以一个Controller响应一组动作绝对是Controller界的真理,Struts的DispatchAction同样可以做到这点。

<action path="/admin/user" name="userForm" scope="request" parameter="method" validate="false">
    
<forward name="list" path="/admin/userList.jsp"/>
action>

    其中parameter="method" 设置了用来指定响应方法名的url参数名为method,即/admin/user.do?method=list 将调用UserAction的public ActionForward list(....) 函数。   

    public ActionForward unspecified(....) 函数可以指定不带method方法时的默认方法。

2. struts-config.xml使用通配符

     对一些CRUD的Action,可以使用通配符,如果这批action里某个action某个方法还存在特殊路径,可以在代码里直接new ActionForward("/foo.jsp");

<action path="/admin/*" name="{1}Form" parameter="method" scope="request" validate="false">
            
<forward name="list" path="/WEB-INF/pages/admin/{1}List.jsp"/>
            
<forward name="edit" path="/WEB-INF/pages/admin/{1}Form.jsp"/>
            
<forward name="success" path="/admin/{1}.do?method=list" redirect="true"/>
action>

3.No FormBean

    Struts 1.2.7 之后一共有三种方法,不需要定义FormBean Java类也不需要在struts-config.xml里详细定义formBean的每个属性。

    第一种是appfuse里使用的,定义DynaValidatorForm里,内嵌一个pojo属性.

<form-bean name="userForm" type="org.apache.struts.validator.DynaValidatorForm">
            
<form-property name="user" type="org.appfuse.model.User"/>
form-bean>

 

//取得form DynaActionForm userForm = (DynaActionForm) form;
User user = (User) userForm.get("user");
//设置form
userForm.set("user",user);

    第二种是用BeanValidatorForm,直接把Pojo作FormBean,无需继承于FormBean接口。

 <form-bean name="userForm" type="org.springside.helloworld.domain.User"/>

 

//取得form Bean
ValidatorForm userForm = (BeanValidatorForm) form;User user = (User) userForm.getInstance();
//设置form
BeanUtils.copyProperties(userForm, user);
    第3种是用LazyValidatorForm ,特别适合于FormBean中并不包含POJO商业对象所有属性的情况,因为通常项目里都属于这种情况,所以springside默认使用lazyValidatorForm.

    比如User对象 有 id,name,status三个属性,而form表单中只有id和name两个input框,如果使用方法1,2,直接把user 作为 form bean,  user对象的status因为没设值,将为null,  copy 到作为商业对象的user时,就会以null覆盖原值。而lazyBean就没有这个问题,如果form中没有status属性,它不会将它copy给商业对象。

    另外,apache commons-beantuils 1.7.0的lazybean仍有小bug,需要使用commons-1.7.1 snapshot版,而且它直接提供下载的snapshot版缺少了几个class,应使用springside提供的版本。

<form-bean name="bookForm" type="org.apache.struts.validator.LazyValidatorForm"/>
 
BeanUtils.copyProperties(user, form);
     
 注意User对象被自动绑定,默认Integer id 在空值时会被赋值为0,需要增加converter,让其默认为null,虽然也可以在web.xml里进行相关配置,但还是在基类里配置了比较安全。
static {
ConvertUtils.register(
new IntegerConverter(null), Integer.class);
}

  4. 一些不必受困于Struts机制的简化写法

    一些简便直接的方法,大家可不必太受困于Struts的机制:

  4.1 不在struts-config.xml配置jsp路径,直接在代码里跳转

return new ActionForward("/foo/bar.jsp");
or
return new ActionForward("/foo/bar.jsp",true);

 

   4.2 不走jsp,直接输出字符串

ActionForward execute(....){
    
try {
            response.setContentType(
"text/html;charset=UTF-8");
            response.getWriter().write(text);
    } 
catch (IOException e) {
            log.error(e);
    }
    
return null;
}

 

 4.3. 不依赖Struts Taglib,EL直接输出FormBean 属性

普通FormBean:${bookForm.image}

LazyValidatorForm:${bookForm.map.image} 

其中bookForm 是formBean名。另一种输出属性的方式是使用jodd-form, 设<jodd:form bean="bookForm"></jodd:form>

4.4 不用i18N地使用Message与Error

   Struts的ActionMessages很常用,但只接收i18N的key作为输入参数,老定义i18n很麻烦,偏方之一就是设一个message= {0},然后直接new ActionMessage("message", message);

5. 终极化简--StrutsEntityAction

 SpringSide封装的EnttiyCRUDAction,,每个包含了CRUD操作的Action仅需如下代码,详细请阅读SpringSide的Struts封装

public class UserAction extends StrutsEntityAction<User,UserManager> {
     
private UserManager userManager;
     
public void setUserManager(UserManager userManager) {
           
this.userManager = userManager;
     }
   }
<jodd:form bean="bookForm"></jodd:form>
   发表时间:2006-11-17  
有意思,我第一次接触struts就觉得强制的每个操作一个Action的写法不爽,应该将对同一个表的操作封装在一个Action类里。
现在真的支持这种用法了,不过我早自己改造了一下以支持此用法了。
0 请登录后投票
   发表时间:2006-11-18  
以前也不是说非要每一个操作对应一个Action啊!
可以在页面中加一个操作参数,根据参数的取值在一个Action中进行不同的操作!
0 请登录后投票
   发表时间:2006-11-18  
那种if-else 比较痛苦,还是直接映射得了。

本贴都沉到底了,终于有兄弟顶了一下,谢谢阿。

各位和Struts作战了几年的兄弟,有什么奇技淫巧分享下吧。
0 请登录后投票
   发表时间:2006-11-18  
To 江南白衣:
呵呵!新版本确实比以前方便多了!
论坛里面技术太多,很多自己都没听说过,不过看到自己做项目时第一次用到的框架--struts,能不顶一下吗?呵呵!

从硬着头皮看Rod的《J2EE development without EJB》开始,觉得spring确实是好东西,SpringMVC就不错,有空的时候,老兄发一篇SpringMVC开发的帖子吧!小弟也学习一下;有空也看看你们的SpringSide!

“奇技淫巧”?  呵呵!

0 请登录后投票
   发表时间:2006-11-19  
请问白衣兄,所指的“lazybean仍有小bug”
是不是copy后null值变成了java.lang.obj[34235]的问题?
0 请登录后投票
   发表时间:2006-11-19  
在1.1里也可以不写FormBean的吧?只要把空标签<form-beans>写上就行了。

dispatchaction还是有一点作用的。
formbean已经越来越显得没用了。
0 请登录后投票
   发表时间:2006-11-19  
liuyxit 写道
请问白衣兄,所指的“lazybean仍有小bug”
是不是copy后null值变成了java.lang.obj[34235]的问题?


是的,1.7.1-snapshot.jar已修复该问题,但snapshot.jar又没有编译optional/org/apache/ 下的几个类,springside作了修复。
0 请登录后投票
   发表时间:2006-11-20  
江南白衣 写道
liuyxit 写道
请问白衣兄,所指的“lazybean仍有小bug”
是不是copy后null值变成了java.lang.obj[34235]的问题?


是的,1.7.1-snapshot.jar已修复该问题,但snapshot.jar又没有编译optional/org/apache/ 下的几个类,springside作了修复。


感谢,已备存springSide的jar。
0 请登录后投票
   发表时间:2006-11-20  
struts毕竟老了,用struts2或jsf吧。JSF也可以做的很pragmatic的
0 请登录后投票
论坛首页 Java企业应用版

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