当前位置导航:炫浪网>>网络学院>>编程开发>>JAVA教程>>Java进阶

用ActionMapping构建漂亮的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>元素及其子元素:<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.str
相关内容
赞助商链接