MILY: 宋体; mso-ascii-font-family: 'Times New Roman'; mso-hansi-font-family: 'Times New Roman'">用ActionMapping构建漂亮的Struts应用程序
在本系列指南的第四部分,你可以学习如何运用ActionMapping来改进你的Struts应用程序。
by Budi Kurniawan
阅读本系列指南的前三部分:
第一部分:“你的第一个Struts应用程序”
第二部分:“Struts应用程序中的流程控制”
第三部分:“用ActionForward优化你的Struts应用程序”
邱吉尔曾经说过他喜欢学习新东西,但他觉得没有必要让别人教他。不管你是否喜欢学习新知识,也不管有人教你还是你自学,学习Java通常意味着仔细研究许多特殊的类。学习Struts也是如此。
这就是我写本系列第四部分的初衷,在本文中,我将详述org.apache.struts.action.ActionMapping类,它是从org.apache.struts.config.ActionConfig派生的。ActionMapping将一个请求路径映射到一个action类,它是Struts应用程序中最常用的类之一。在你深入学习这个类时,你会重新用到在该系列第1、2和3部分创建的两个login应用程序,从而了解如何运用ActionMapping来重写应用程序。
当然,你可能不记得以前用过任何ActionMapping实例了。这是因为控制器servlet实际上为你创建了它们。你只需要配置在Struts配置文件(struts-config.xml)中创建的每个ActionMapping实例就行了(通过给它的属性赋值)。了解这些属性对正确运用ActionMapping类很重要,因此我将讲述这些属性以及如何定义它们。
首先,让我们回想一下,Struts配置文件的根元素是<struts-config>。<struts-config>元素可以包含一个可选的<action-mappings>元素,同样<action-mappings>元素可以包含<action>元素。例如,下面就是本系列第三部分的login应用程序配置文件中的<struts-config>元素及其子元素:
clearcase/" target="_blank" >cc"><struts-config> <action-mappings> <action path="/login" type="com.javapro.struts.LoginAction"/> <action path="/logout" type="com.javapro.struts.LogoutAction"/> <action path="/viewSecret" type="com.javapro.struts.ViewSecretAction"/> </action-mappings> </struts-config> |
<action-mapping>中的每个<action>都代表控制器servlet创建的一个ActionMapping实例。一个<action>元素可以包含多个特性,每个特性都和ActionMapping实例中的一个属性相应。
作为例子,我们来看前面的Struts配置文件中的这个<action>元素:
<action path="/login" type="com.javapro.struts.LoginAction"/> |
这个<action>将路径“/login”映射到action类com.javapro.struts.LoginAction。换句话说,一个以“/login.do”结尾的用户请求将被传递到LoginAction类。然而,ActionMapping也有其它的用途。你(Struts程序员)可以通过给它的属性赋值来给ActionMapping实例下达action指令。(有些属性与action forms相关,我将在本系列的第五部分讲述。)
ActionMapping类的属性
ActionMapping有许多属性。首先,它从ActionConfig类继承了一些属性——如type、forward、include和unknown。它们与action forms是无关的。前三个属性是ActionMapping类最重要的属性。你只能指定其中的一个,所以,如果一个<action>元素已经定义了一个type属性,它就不能有forward属性或include属性了。
type属性的值是路径所映射的Action类的完全限定的Java类的名称。(你曾在前面的login应用程序中的Struts配置文件中用过这个属性。)如果运用了type属性,那么控制器servlet就可以调用action实现类的execute方法,传递恰当的ActionMapping实例。注意,org.apache.struts.action.Actionclass类的execute方法有如下的定义(第一个参数是一个ActionMapping实例):
public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException |
forward属性代表的是满足该请求的context-relative资源(通过调用RequestDispatcher.forward()),而不是实例化type属性指定的Action类。(更多关于如何运用这个属性的信息,参见后面的‘修改login应用程序’部分。)
我们给include属性赋的值是满足该请求的context-relative资源路径(通过调用RequestDispatcher.include()),而不是实例化type属性指定的Action类。
注意,<action>元素包含的forward属性与可能出现在<action>元素下的<forward>元素是不同的。这就是说,一个<action>元素可以有一个type属性,以及一个或多个<forward>子元素,如下面这个<action>元素所示(我马上会讲述<forward>元素):
<action path="/login" type="com.javapro.struts.LoginAction"> <forward name="success" path="/mainMenu.jsp"/> <forward name="failure" path="/login.jsp"/> </action> |
我们用<path>属性来指定这个ActionMapping将处理的请求路径。最后,用unknown属性来处理未知的路径。在一个action元素中,将这个属性设置为true,使这个action成为该应用程序缺省的action。换句话说,它处理所有其它的action不能处理的请求。在一个单独的应用程序中,只有一个action可以被定义成是缺省的。
例如,下面这个<action>元素将unknown属性设置为true,使该action成为缺省的:
<action path="/login" type="com.javapro.struts.LoginAction" unknown="true"/> |
<forward>元素
<forward>元素描述了一个逻辑名称与一个context-relative URI路径识别的资源之间的映射。它有以下这些属性:
|
· className 这是你想运用的ActionForward实现类的完全限定的Java类的名称。缺省情况下,它的值是作为“forward”初始化参数给Struts控制器servlet配置的。 |
|
· contextRelative 在一个模块化应用程序中,如果路径属性是以一个斜线(“/”)开头的,并且是相对于整个Web应用程序的,而不是相对于该模块的,我们就将这个属性设置为true。缺省情况下是false。 |
|
· name 这是forward的唯一标识符,用来在应用程序的action类中引用它。 |
|
· path 被映射资源的context-relative路径。 |
|
· redirect 设置成true,运用sendRedirect()引导到该资源;或者设置成false,运用RequestDispatcher.forward()作为替代。 |
运用<action>下的<forward>元素意味着你不必在你的ActionForward对象中写死路径名。例如,我们来看前面的login应用程序中LoginAction类的execute方法中的代码:
public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { String userName = request.getParameter("userName"); String password = request.getParameter("password"); if (userName!=null && password!=null && userName.equals("john") && password.equals("123")) { HttpSession session = request.getSession(); session.setAttribute("loggedIn", "1"); return (new ActionForward("/mainMenu.jsp")); } else { return (new ActionForward("/login.jsp")); } } |
注意,最后的两个return语句写死了mainMenu.jsp和login.jsp页面。如果任意一个文件名发生改变,你都必须重新编译LoginAction类。但是如果你用<forward>元素,你可以用一个名称映射mainMenu.jsp页面,用另一个名称映射login.jsp页面。现在,如果你需要改变文件名,你就可以在配置文件中进行改变,而不需要重新编译了。要这么做,你需要在struts-config.xml文件中声明这个<action>元素:
<action path="/login" type="com.javapro.struts.LoginAction"> <forward name="success" path="/mainMenu.jsp"/> <forward name="failure" path="/login.jsp"/> </action> |
现在,mainMenu.jsp就与“success”这个名称联系在一起了,login.jsp与“failure”联系在一起了。你可以通过调用ActionMapping类的findForward方法,传入相关的名称从一个action实现类的内部得到ActionForward实例:
mapping.findForward(name); |
例如,要得到包含路径“/mainMenu.jsp”的ActionForward对象,我们可以用下面的方法:
mapping.findForward("success"); |
同样,要得到包含路径“/login.jsp”的ActionForward对象,可以用:
writemapping.findForward("failure"); |
接下来,我们该用<forward>方法重写login应用程序了。
修改Login应用程序
注意,你在第三部分创建的login应用程序发生了怎样的变化。首先,我们来看原应用程序中ViewSecretAction类的execute方法:
public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { return (new ActionForward("/viewSecret.jsp")); } |
除了返回一个路径是“/viewSecret.jsp”的ActionForward对象外,该execute方法没有做任何的处理。通过运用struts-config.xml文件中<action>元素的forward属性,我们就不需要ViewSecretAction类了。要实现这一点,我们可以将下面的<action>代码:
<action path="/viewSecret" type="com.javapro.struts.ViewSecretAction"/> |
替换成:
<action path="/viewSecret" forward="/viewSecret.jsp"/> |
现在,我们就用forward属性替换了type属性。然后,通过添加unknown属性并将它设置成true,我们就可以使login页面成为缺省的页面。实际上,任何未知路径(以.do结尾)都会被引导到LoginAction实例。处理“/login”的<action>元素如下所示:
<action path="/login" type="com.javapro.struts.LoginAction" unknown="true"> |
对LoginAction和LogoutAction类所做的最后的修改运用了ActionMapping类的findForward方法来得到适当的ActionForward对象。在struts-config.xml文件中声明映射(见列表1)。另外,我们也来看看修改过了的LoginAction类(见列表2)和修改过了的LogoutAction类(见列表3)。
注意,该版本运用了很少的action实现类,因为我们不再需要ViewSecretAction了。另一个好处是:这种方法避免了在action实现类中写死路径名。另外,/login路径现在是缺省的了。结果就是:你的Struts代码最终看上去像一个真正的Struts应用程序了。你在学习了第五部分(关于action forms)后,这个程序会变得更好