Action 只是Struts 2控制器的一部分,只负责处理用户请求。当Action处理完请求后,处理的结果会通过视图资源来展示,此时通过<result>元素配置逻辑视图和物理视图之间的映射关系。
result处理流程
Action处理完请求后,将返回一个字符串,整个字符串就是一个逻辑视图名。该字符串在struts.xml配置文件中对应了一个物理视图资源,通常就是JSP页面。Struts 2框架通过配置文件中<action>的<result>子元素配置逻辑视图名和物理视图资源之间的映射关系。当Struts 2框架收到Action返回的某个逻辑视图名时,就会将对应的物理视图呈现给用户。
Struts 2框架的result处理流程如下:
- 用户发送请求;
- Struts 2框架将用户请求转发到Action控制器;
- Action控制器处理完用户请求后,返回一个逻辑视图(普通的字符串);
- Struts 2收到这个逻辑视图后,将请求转发到对应的视图资源;
- 视图资源将处理结果呈现给用户。
配置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是最常用的结果类型。
- 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>
- 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页面。