Osheep

时光不回头,当下最重要。

Struts 2 result详解

Action 只是Struts 2控制器的一部分,只负责处理用户请求。当Action处理完请求后,处理的结果会通过视图资源来展示,此时通过<result>元素配置逻辑视图和物理视图之间的映射关系。

result处理流程

Action处理完请求后,将返回一个字符串,整个字符串就是一个逻辑视图名。该字符串在struts.xml配置文件中对应了一个物理视图资源,通常就是JSP页面。Struts 2框架通过配置文件中<action>的<result>子元素配置逻辑视图名和物理视图资源之间的映射关系。当Struts 2框架收到Action返回的某个逻辑视图名时,就会将对应的物理视图呈现给用户。

Struts 2框架的result处理流程如下:

  1. 用户发送请求;
  2. Struts 2框架将用户请求转发到Action控制器;
  3. Action控制器处理完用户请求后,返回一个逻辑视图(普通的字符串);
  4. Struts 2收到这个逻辑视图后,将请求转发到对应的视图资源;
  5. 视图资源将处理结果呈现给用户。
配置result

逻辑视图和物理视图之间的映射是通过在struts.xml中配置<result>元素来实现。配置result是告诉Struts 2框架当Action处理结束后,系统下一步要做什么,应该调用哪个视图资源来显示处理结果。

根据<result>元素在struts.xml文件中所在位置的不同,可以将result分为以下两种:

  • 局部result —— 将<result>元素作为<action>元素的子元素配置。
  • 全局result —— 将<result>元素作为<global-results>元素的子元素配置。
局部result

一个Action可以对应多个result。局部result只在特定的Action范围内有效,一个Action不能使用另外一个Action配置的局部result。

配置<result>元素需要指定name和type属性:

  • name属性指定逻辑视图名称,即Action返回的字符串。
  • type 属性指的是result类型,默认值是dispatcher,表示请求转发到JSP页面。

局部result示例

<package name="default" namespace="/" extends="struts-default">
    <action name="login" class="com.struts.example.action.LoginAction">
        <result name="success" type = "dispatcher">/ok.jsp</result>
    </action>
</package>

配置<result>元素时如果没有指定name和type属性值,系统将使用默认值,其中success是默认的name属性值,dispatcher是默认的type属性值。因此上面的示例代码可以简化成这样:

<package name="default" namespace="/" extends="struts-default">
    <action name="login" class="com.struts.example.action.LoginAction">
        <result>/ok.jsp</result>
    </action>
</package>
全局result

全局result在<global – results>元素中指定,全局result的作用范围是对所有的Action。

全局result示例:

<package name="default" namespace="/" extends="struts-default">
    <global - results>
        <result>/ok.jsp</result>
    </global - results>
     <action name="login" class="com.struts.example.action.LoginAction" />
</package>

如果一个Action中包含了与全局result同名的局部result,则局部result会覆盖全局result。当Action处理完用户请求后,首先搜索当前Action中的局部result,匹配失败时才会搜索全局result。

result类型

针对不同的视图技术,Struts 2提供了一系列的结果类型,如下表所示:

结果类型 描述
chain 用于进行Action链式处理
chart 用于整合JFreeChart技术
dispatcher 用于整合JSP技术
freemarker 用于整合FreeMarker技术
httpheader 用于控制特殊的HTTP行为
jasper 用于整合JasperReport报表技术
jsf 用于整合JSF技术
redirect 用于重定向到其他的URL
redirectAction 用于重定向到其他的Action
stream 用于向浏览器返回InputStream,一般用于文件下载
tiles 用于整合Tiles技术
velocity 用于整合Velocity技术
xslt 用于整合XML/XSLT技术
plainText 用于显示某个页面的源代码

其中dispatcher、redirect和redirectAction是最常用的结果类型。

  1. redirect类型

redirect类型和dispatcher类型类似,dispatcher类型是将请求转发到指定的JSP资源;而redirect类型将请求重定向到指定的视图资源。

dispatcher和redirect之间的区别主要是转发和重定向的区别:重定向会丢失所有的请求参数、请求属性,同时Action的处理结果也会丢失。当使用redirect类型时,系统实际上会调用HttpServletResponse的sendRedirect()方法来重定向指定视图资源,这种重定向的效果就是产生一个新的请求。

    <package name="test" extends = "struts-default">
        <action name = "login" class = "com.chasel.result.action.loginAction">
            <result type = "redirect">/index.jsp</result>
        </action>
    </package>
  1. redirectAction类型

redirectAction类型和redirect类型相似,都是生成一个全新的请求,都会丢失请求参数、请求属性和前一个Action的处理结果。与redirect的区别是在于:redirectAction使用ActionMapperFactory提供的ActionMapper进行重定向;而redirect使用HttpServletResponse的sendRedirect()方法来重定向。

当需要让一个Action处理结束后,直接将请求重定向到另一个Action时,可以使用redirectAction类型。

配置redirectAction类型需要指定actionName和namespace两个参数

    <package name="product" extends = "struts-default">
        <action name = "* product" class = "com.chasel.result.action.ProductAction" method = {1}>
            <result name = "list">/productList.jsp</result>
            <result name = "edit">/edit.jsp</result>
            <result name = "del">/del.jsp</result>
        </action>
    </package>
    <package name="user" extends = "struts-default">
        <action name = "login" class = "com.chasel.result.action.loginAction">
            <result type = "redirectAction">
                <param name="actionName"> listproduct</param>
                <param name="namespace">/product</param>
            </result>
        </action>
    </package>

其中actionName参数指定重定向的Action名称为listproduct,namespace参数指定需要重定向的Action所在的命名空间。

动态result

动态result是指在指定实际视图资源时使用表达式语法,通过这种语法实现动态转入实际的视图资源。

在介绍配置Action时,讲到可以使用通配符来配置动态Action。同样,在配置<result>元素时也可以使用{N}表达式,从而根据请求动态决定实际资源。

    <package name="product" extends = "struts-default">
        <action name = "* product" class = "com.chasel.result.action.ProductAction" method = {1}>
            <result>/{1}</result>
        </action>
    </package>

上述配置中,有一个名为”* product”的Action,该Action处理匹配所有”* product.action”请求,并根据请求动态转入不同的JSP页面。

点赞