精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (3)
|
|
---|---|
作者 | 正文 |
发表时间:2009-05-21
最后修改:2009-05-27
<html:checkbox property="booleanProperty" /> 用来显示单个的选择框,通过对该标签初始化的操作,可以看出许多隐藏在Struts框架“背后”的故事。 假定流程是这样: 1. Jsp页面显示所有用户的列表,只包含姓名,想看某用户的详细信息,就要点击这个用户的名字,通过hyperlink或其他方式传递一个用户ID到Action类中。 2. 在Action类中,通过传来的ID,连接数据库搜寻出该用户的所有信息,其中有一个表示用户在注册时候选择的是否接受订阅的信息。 3. 然后再由Action传到前台jsp页面将用户的详细信息逐条进行显示。 在前台显示用户是否接受订阅信息在jsp页面中用一个checkbox Struts标签表示。也就是说要根据数据库的信息,来初始化这个checkbox是否被选中的状态。 其实最终生成的html只要如下,包含 “checked” 这条就可以, <input type="checkbox" name="booleanProperty" value="on" checked="checked"> 但由于<html:checkbox/>并没有一个可以表示是否被选中的属性供我们设置,而我们想用jsp scriptlet或EL表达式通过判断在标签内部手动加上这么一条都会报错(比如:<%=checked?”checked”:””%> 其实都不知道把这东西放到哪里合适),那么问题出现了,到底应该怎样根据数据库的用户信息来初始化这个checkbox的选中状态呢? 我们知道,ActionForm在比如用户注册的时候,接受用户输入的信息,然后传递到Action类中以供处理。但如我们这种情况,用户是通过一个超链接到达了Action,根本就没有进行什么输入,又怎么用ActionForm呢? 其实真实的情况是这样,不管你来Action类之前有没有什么表单的输入,只要在struts-config.xml配置了这个action的name属性,那么这个虚无的没有任何用户输入数据的表单ActionForm对象照样会被实例化并传递到Action类中。如: <action path="/tagUse" type="TagAction" name="tagHandlerForm"> 而我们在Action类中可以测试一下,看看这个被传进来ActionForm倒底是什么。 form.getClass().getName() form==null? 事实证明确实有个正如我们在配置文件中指定的ActionForm类对象被传了进来,只不过由于没有前台输入表单的缘故,该对象内的属性都是null而已。 ActionForm的生命周期是从request来到response回去,比如用户注册JSP,数据是从request来,而从数据库取值再返回JSP就是response。只要在这个scope,这个ActionForm就会和jsp页面绑定在一起,而这种FormBean字段与jsp控件之间通过控件property属性的绑定,是Struts内置的,或是说隐性的,不可被其他方式所取代的。这是这种隐性的绑定,解决了checkbox初始化的问题。 我们可以这样做,首先取得确实的ActionForm TagHandlerForm myForm = (TagHandlerForm)form; 然后再将该formbean中的属性一一设定为从数据库取来的值 myForm.setName("abcd"); 这里注意:在Action类中的execute方法 ActionForward execute( ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) 我们用的这个form是从方法中传来的,虽然我们在该方法内对这个参数的引用改变是无效的,但却可以改变引用的内容。所以当设置完了以后,与前台JSP页面在这个request – response 范围内紧密,隐性绑定着的ActionForm的内容,也发生了变化。这样,从Action类response到JSP页面的时候,checkbox就得到了我们期待的初始化。 看一下Struts的官方文档,其中有这么一句关于<html:checkbox />的 NOTE: The underlying property value associated with this field should be of type 说白了就是要你把关于这个checkbox的属性,在ActionForm里设成boolean. 表面上看这只是一个在JSP上显示名字的值罢了 (<html:checkbox property="booleanProperty" />),为什么要设置成boolean,用String不是更直接?但正是因为ActionForm的这种隐式的数据绑定,使我们可以仅用上面的一条语句便将checkbox正确的初始化。 再看一下配置文件 <action path="/tagUse" type="strutsTutorial.PrepareTagAction" name="tagHanderForm" attribute="tagInfo"> attribute的真正意思是“显式”生成一个在指定scope中的对象用于“取代”默认的name 所定义的ActionForm,一旦声明了这个属性(还必须在Action类中将其放入scope才行 request.setAttribute("tagInfo",aTag);),ActionForm本身所具有的那种与JSP紧密而隐式的数据绑定,便被破坏了。所以如果用这种设置attribute的方式返回了JSP页面,控件就必须显式赋值 <html:text property="name" value="${tagInfo.name}"/>,但像checkbox这种没办法显示赋值的标签,它就无能无力了(借助javascript应该也可以)。 而对于我们这个假定的流程来说,更好的做法应该是,准备两个类,TagHandlerForm 和 Tags。TagHandlerForm类只用来收集/显示从jsp页面来/去的信息(即典型的ActionForm类);Tags类用来存储从database取得的信息(应该是一个POJO的数据封装类)。它们其实很相像,拥有近乎相同的属性,可以说前者是后者的子集,但也不是绝对,比如form类包含一个hidden field的属性,用来在Action类的处理逻辑中做以区别,而pojo类其实也没必要存储这个。但pojo类至少应该包括form类需要在jsp用来显示的所有属性。这样,当用户提交表单的时候,把ActionForm的内容复制到pojo类中,然后再完全脱离form类的情况下进行数据处理(也就是说从presentation/control layer转到model layer),反之亦然。这也正是mvc2设计模式希望我们做到的。 所以上面的做法可以再修改一下 Tags aTag = new Tags(); 数据类 BeanUtils.copyProperties(myForm,aTag); 数据转移 request.setAttribute("tagInfo",aTag); 在这里设置一个request scope的属性对象是不会影响到ActionForm的隐性绑定,而且form类中有些值是与JSP控件无法隐性绑定的比如<html:img />标签(通过多加一个数据类对象,我们则可以 <html:img page=”${tagInfo.imgSrc}”> 动态加载属于用户的图片), 或者数据类中我们还有些附加值想带到JSP去。
声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
浏览 1949 次