UML软件工程组织

用Struts框架开发Java应用(2)
作者:赵科 选择自 开放系统世界——赛迪网
注意,LogonForm中的属性名一定要和logon.jsp中的表单域名完全对应起来。Struts就是由此从浏览器端抓取提交的数据,并填充到LogonForm对象中,再传送给LogonAction类进行处理的。Struts实现的表单验证和重填技术也是这样实现的。注意,在上面的页面代码中我们使用了Struts的HTML定制标签库,也可以直接使用类似 <input type="text" name="username" /> 的HTML代码,但这样就失去了自动回填的支持。

Action

现在我们已经对Struts前端应用框架的整体结构、数据流转有了一定的认识。也知道了Struts是怎样从配置文件中获取配置信息的,即先启动,然后等待请求,再从前台抓取数据,根据配置信息调用(或生成)Action类进行处理,最后根据处理的结果转向到对应的页面响应用户。那么,现在就让我们看看LogonAction是怎样进行业务处理,并将处理结果显示给用户的。代码如下:


package org.apache.struts.webapp.example; 
import org.apache.struts.action.Action; 
......
public final class LogonAction extends Action {
 public ActionForward perform(ActionMapping mapping,
  ActionForm form,
  HttpServletRequest request,
  HttpServletResponse response)
throws IOException, ServletException {
......
ActionErrors errors = new ActionErrors();
String username = ((LogonForm) form).getUsername();
String password = ((LogonForm) form).getPassword();
...... 
// 进行用户验证
// 如果不存在此用户,或密码错误,则将错误添加到errors中
if (!errors.empty()) {
saveErrors(request, errors);
return (new ActionForward(mapping.getInput()));
}
HttpSession session = request.getSession();
session.setAttribute(Constants.USER_KEY, user);
//删除过期的form bean
......
if (mapping.getAttribute() != null) {
if ("request".equals(mapping.getScope()))
request.removeAttribute(mapping.getAttribute());
else
session.removeAttribute(mapping.getAttribute());
}
//定位到成功页面
return (mapping.findForward("success"));
  }
 }
}


LogonAction类继承于org.apache.struts.action.Action,Action类是所有的Action的基类。其中定义的perform()方法完成对请求处理,并根据处理结果转向到不同的页面,然后显示给用户。Action类并不是Servlet,它不直接监听来自客户端的请求。上面所提到的中心控制器ActionServlet是一个Servlet,客户端发出的Action请求,由ActionServlet接收,根据struts-config.xml中的配置,传入对应的mapping、form、request、response对象,并调用对应的Action类的perform()方法进行处理(在第一次调用时,实例化一个Action,随后的请求将直接调用已存在的Action类进行处理,所以Action类是共享的,编程中注意处理并发问题)。下面对上述的LogonAction的perform()方法进行简单的分析。

1. 首先从传入的form对象中获取username和password。大家可能会奇怪这两个属性是在什么时候置入的?这是ActionServlet在调用LogonAction的perform()方法之前根据struts-config.xml中的配置及对应的LogonForm中的属性从前端请求抓取数据,置入LogonForm对象的。然后进行用户验证。本例没有使用数据库进行存储,用户信息是存储在database.xml文件中的。如果username/password在database.xml文件不存在或密码错误,则生成一个ActionError对象,将错误信息存储到该对象中,并把这个对象添加到ActionErrors中,其它发生的错误也类似处理。在随后的程序中,首先校检是否有错误发生,如果有错误发生,就把错误对象存储到request中,使用的方法为saveErrors(request, errors)。该方法是在基类org.apache.struts.action.Action中实现的,它完成的功能很简单,将errors对象通过setAttribute()方法存储到request中即可。实现如下:


protected void saveErrors(HttpServletRequest request,ActionErrors errors) {
 //删除掉不需要的错误信息
 if ((errors == null) || errors.empty()) {
  request.removeAttribute(ERROR_KEY);
  return;
 }
 //保存我们需要的错误对象
 request.setAttribute(ERROR_KEY, errors);
}


如果发生错误,将重定向到输入页面,同时自动完成输入域的重填,代码如下:


return (new ActionForward(mapping.getInput()));


如果在图5所示页面输入user/abc提示上面的错误,那么原来输入的username域就会自动填充进来了。在一个有大量输入域的表单会大大方便用户。



图5 登陆失败页面


2. 在上面的程序执行通过后,就说明这是一个合法登陆。我们要做的就是把当前用户保存到session中,同时清除已经无效的Form Bean,最后转向到success页面,代码为:


return (mapping.findForward("success"))。


一个ActionMapping对象对应于struts-config.xml中的一个 <action/> 的配置。细心的读者可能已经发现了,在struts-config.xml中举例的/logon Action的配置中,并没有Forward配置,但在它的开始部分却有一个全局的Forward配置。配置代码如下:


<global-forwards>
<forward   name="logoff"  path="/logoff.do"/>
<forward   name="logon"   path="/logon.jsp"/>
<forward   name="success" path="/mainMenu.jsp"/>
</global-forwards>


这个Forward配置将对所有的Action都有效。但如果 中已经有和这些全局Forward配置同名的项,那么它将覆盖全局配置。所以你登录成功后,将转向到/mainMenu.jsp页面。

 

 

版权所有:UML软件工程组织