4.11、AbstractWizardFormController
向导控制器类提供了多步骤(向导)表单的支持(如完善个人资料时分步骤填写基本信息、工作信息、学校信息等)
假设现在做一个完善个人信息的功能,分三个页面展示:
1、页面1完善基本信息;
2、页面2完善学校信息;
3、页面3完善工作信息。
这里我们要注意的是当用户跳转到页面2时页面1的信息是需要保存起来的,还记得AbstractFormController中的sessionForm吗? 如果为true则表单数据存放到session中,哈哈,AbstractWizardFormController就是使用了这个特性。
向导中的页码从0开始;
PARAM_TARGET = "_target":
用于选择向导中的要使用的页面参数名前缀,如“_target0”则选择第0个页面显示,即图中的“wizard/baseInfo”,以此类推,如“_target1”将选择第1页面,要得到的页码为去除前缀“_target”后的数字即是;
PARAM_FINISH = "_finish":
如果请求参数中有名为“_finish”的参数,表示向导成功结束,将会调用processFinish方法进行完成时的功能处理;
PARAM_CANCEL = "_cancel":
如果请求参数中有名为“_cancel”的参数,表示向导被取消,将会调用processCancel方法进行取消时的功能处理;
向导中的命令对象:
向导中的每一个步骤都会把相关的参数绑定到命令对象,该表单对象默认放置在session中,从而可以跨越多次请求得到该命令对象。
接下来具体看一下如何使用吧。
(1、修改我们的模型数据以支持多步骤提交:
1
2
3
4
5
6
7
8
|
public class UserModel {
private String username;
private String password;
private String realname; //真实姓名
private WorkInfoModel workInfo;
private SchoolInfoModel schoolInfo;
//省略getter/setter
} |
1
2
3
4
5
6
|
public class SchoolInfoModel {
private String schoolType; //学校类型:高中、中专、大学
private String schoolName; //学校名称
private String specialty; //专业
//省略getter/setter } |
1
2
3
4
5
6
|
public class WorkInfoModel {
private String city; //所在城市
private String job; //职位
private String year; //工作年限
//省略getter/setter } |
(2、控制器
-
package cn.javass.chapter4.web.controller;
-
//省略import
-
public class InfoFillWizardFormController extends AbstractWizardFormController {
-
public InfoFillWizardFormController() {
-
setCommandClass(UserModel.class);
-
setCommandName("user");
-
}
-
protected Map referenceData(HttpServletRequest request, int page) throws Exception {
-
Map map = new HashMap();
-
if(page==1) { //如果是填写学校信息页 需要学校类型信息
-
map.put("schoolTypeList", Arrays.asList("高中", "中专", "大学"));
-
}
-
if(page==2) {//如果是填写工作信息页 需要工作城市信息
-
map.put("cityList", Arrays.asList("济南", "北京", "上海"));
-
}
-
return map;
-
}
-
protected void validatePage(Object command, Errors errors, int page) {
-
//提供每一页数据的验证处理方法
-
}
-
protected void postProcessPage(HttpServletRequest request, Object command, Errors errors, int page) throws Exception {
-
//提供给每一页完成时的后处理方法
-
}
-
protected ModelAndView processFinish(HttpServletRequest req, HttpServletResponse resp, Object command, BindException errors) throws Exception {
-
//成功后的处理方法
-
System.out.println(command);
-
return new ModelAndView("redirect:/success");
-
}
-
protected ModelAndView processCancel(HttpServletRequest request, HttpServletResponse response, Object command, BindException errors) throws Exception {
-
//取消后的处理方法
-
System.out.println(command);
-
return new ModelAndView("redirect:/cancel");
-
}
-
}
page页码:是根据请求中以“_target”开头的参数名来确定的,如“_target0”,则页码为0;
referenceData:提供每一页需要的表单支持对象,如完善学校信息需要学校类型,page页码从0开始(而且根据请求参数中以“_target”开头的参数来确定当前页码,如_target1,则page=1);
validatePage:验证当前页的命令对象数据,验证应根据page页码来分步骤验证;
postProcessPage:验证成功后的后处理;
processFinish:成功时执行的方法,此处直接重定向到/success控制器(详见CancelController);
processCancel:取消时执行的方法,此处直接重定向到/cancel控制器(详见SuccessController);
其他需要了解:
allowDirtyBack和allowDirtyForward:决定在当前页面验证失败时,是否允许向导前移和后退,默认false不允许;
onBindAndValidate(HttpServletRequest request, Object command, BindException errors, int page):允许覆盖默认的绑定参数到命令对象和验证流程。
(3、spring配置文件(chapter4-servlet.xml)
1
2
3
4
5
6
7
8
9
10
|
< bean name = "/infoFillWizard" class = "cn.javass.chapter4.web.controller.InfoFillWizardFormController" >
< property name = "pages" >
< list >
< value >wizard/baseInfo</ value >
< value >wizard/schoolInfo</ value >
< value >wizard/workInfo</ value >
</ list >
</ property >
</ bean >
|
pages:表示向导中每一个步骤的逻辑视图名,当InfoFillWizardFormController的page=0,则将会选择“wizard/baseInfo”,以此类推,从而可以按步骤选择要展示的视图。
(4、向导中的每一步视图
(4.1、基本信息页面(第一步) baseInfo.jsp:
1
2
3
4
|
< form method = "post" >
真实姓名:< input type = "text" name = "realname" value = "${user.realname}" >< br />
< input type = "submit" name = "_target1" value = "下一步" />
</ form >
|
当前页码为0;
name="_target1":表示向导下一步要显示的页面的页码为1;
(4.2、学校信息页面(第二步) schoolInfo.jsp:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
< form method = "post" >
学校类型:< select name = "schoolInfo.schoolType" >
< c:forEach items = "${schoolTypeList }" var = "schoolType" >
< option value = "${schoolType }" <c:if test = "${user.schoolInfo.schoolType eq schoolType}" >
selected="selected"
</ c:if >
>
${schoolType}
</ option >
</ c:forEach >
</ select >< br />
学校名称:< input type = "text" name = "schoolInfo.schoolName" value = "${user.schoolInfo.schoolName}" />< br />
专业:< input type = "text" name = "schoolInfo.specialty" value = "${user.schoolInfo.specialty}" />< br />
< input type = "submit" name = "_target0" value = "上一步" />
< input type = "submit" name = "_target2" value = "下一步" />
</ form >
|
(4.3、工作信息页面(第三步) workInfo.jsp:
-
<form method="post">
-
所在城市:<select name="workInfo.city">
-
<c:forEach items="${cityList }" var="city">
-
<option value="${city }"
-
<c:if test="${user.workInfo.city eq city}">selected="selected"</c:if>
-
>
-
${city}
-
</option>
-
</c:forEach>
-
</select><br/>
-
职位:<input type="text" name="workInfo.job" value="${user.workInfo.job}"/><br/>
-
工作年限:<input type="text" name="workInfo.year" value="${user.workInfo.year}"/><br/>
-
<input type="submit" name="_target1" value="上一步"/>
-
<input type="submit" name="_finish" value="完成"/>
-
<input type="submit" name="_cancel" value="取消"/>
-
</form>
当前页码为2;
name="_target1":上一步,表示向导上一步要显示的页面的页码为1;
name="_finish":向导完成,表示向导成功,将会调用向导控制器的processFinish方法;
name="_cancel":向导取消,表示向导被取消,将会调用向导控制器的processCancel方法;
到此向导控制器完成,此处的向导流程比较简单,如果需要更复杂的页面流程控制,可以选择使用Spring Web Flow框架。
4.12、ParameterizableViewController
参数化视图控制器,不进行功能处理(即静态视图),根据参数的逻辑视图名直接选择需要展示的视图。
1
2
3
4
|
< bean name = "/parameterizableView" class = "org.springframework.web.servlet.mvc.ParameterizableViewController" >
< property name = "viewName" value = "success" />
</ bean >
|
该控制器接收到请求后直接选择参数化的视图,这样的好处是在配置文件中配置,从而避免程序的硬编码,比如像帮助页面等不需要进行功能处理,因此直接使用该控制器映射到视图。
4.13、AbstractUrlViewController
提供根据请求URL路径直接转化为逻辑视图名的支持基类,即不需要功能处理,直接根据URL计算出逻辑视图名,并选择具体视图进行展示:
urlDecode:是否进行url解码,不指定则默认使用服务器编码进行解码(如Tomcat默认ISO-8859-1);
urlPathHelper:用于解析请求路径的工具类,默认为org.springframework.web.util.UrlPathHelper。
UrlFilenameViewController是它的一个实现者,因此我们应该使用UrlFilenameViewController。
4.14、UrlFilenameViewController
将请求的URL路径转换为逻辑视图名并返回的转换控制器,即不需要功能处理,直接根据URL计算出逻辑视图名,并选择具体视图进行展示:
根据请求URL路径计算逻辑视图名;
-
<bean name="/index1/*"
-
class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>
-
<bean name="/index2/**"
-
class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>
-
<bean name="/*.html"
-
class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>
-
<bean name="/index3/*.html"
-
class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>
/index1/*:可以匹配/index1/demo,但不匹配/index1/demo/demo,如/index1/demo逻辑视图名为demo;
/index2/**:可以匹配/index2路径下的所有子路径,如匹配/index2/demo,或/index2/demo/demo,“/index2/demo”的逻辑视图名为demo,而“/index2/demo/demo”逻辑视图名为demo/demo;
/*.html:可以匹配如/abc.html,逻辑视图名为abc,后缀会被删除(不仅仅可以是html);
/index3/*.html:可以匹配/index3/abc.html,逻辑视图名也是abc;
上述模式为Spring Web MVC使用的Ant-style 模式进行匹配的:
-
? 匹配一个字符,如/index? 可以匹配 /index1 , 但不能匹配 /index 或 /index12
-
* 匹配零个或多个字符,如/index1/*,可以匹配/index1/demo,但不匹配/index1/demo/demo
-
** 匹配零个或多个路径,如/index2/**:可以匹配/index2路径下的所有子路径,如匹配/index2/demo,或/index2/demo/demo
-
-
如果我有如下模式,那Spring该选择哪一个执行呢?当我的请求为“/long/long”时如下所示:
-
/long/long
-
/long/**/abc
-
/long/**
-
/**
-
Spring的AbstractUrlHandlerMapping使用:最长匹配优先;
-
如请求为“/long/long” 将匹配第一个“/long/long”,但请求“/long/acd” 则将匹配 “/long/**”,如请求“/long/aa/abc”则匹配“/long/**/abc”,如请求“/abc”则将匹配“/**”
UrlFilenameViewController还提供了如下属性:
prefix:生成逻辑视图名的前缀;
suffix:生成逻辑视图名的后缀;
1
2
3
|
protected String postProcessViewName(String viewName) {
return getPrefix() + viewName + getSuffix();
} |
1
2
3
4
|
< bean name = "/*.htm" class = "org.springframework.web.servlet.mvc.UrlFilenameViewController" >
< property name = "prefix" value = "test" />
< property name = "suffix" value = "test" />
</ bean >
|
当prefix=“test”,suffix=“test”,如上所示的/*.htm:可以匹配如/abc.htm,但逻辑视图名将变为testabctest。
相关推荐
总的来说,Controller接口控制器是Spring MVC中连接用户请求和业务逻辑的关键组件。通过合理使用@Controller、@RequestMapping等注解,我们可以创建出灵活、可扩展的Web服务。深入学习和理解Controller接口及其工作...
第四章 Controller接口控制器详解(1).pdf 第四章 Controller接口控制器详解(2).pdf 第四章 Controller接口控制器详解(3).pdf 第四章 Controller接口控制器详解 (4).pdf 第四章 Controller接口控制器详解 (5...
本代码使用了Spring MVC框架(spring2.5架包) 演示了(Controller接口的试用方法)和 MultiActionController多动作控制器 数据库连接试用Spring JDBC 并且着重介绍了MultiActionController多动作控制器的两种方法名...
getControllerName() 和 setControllerName() 方法用于获取和设置控制器名称;getActionName() 和 setActionName() 方法用于获取和设置动作名称。此外,还提供了获取和设置模块、控制器、动作对应的键名的方法。 在...
- `spi_controller_driver`定义了SPI控制器驱动的接口函数,包括初始化、配置、传输等。 - `spi_register_driver`用于注册SPI设备驱动,`spi_unregister_driver`用于注销。 3. **SPI控制器驱动** - 控制器驱动...
USB EHCI主机控制器详解 USB(通用串行总线)技术是计算机和电子设备之间进行数据交换的标准通信协议。EHCI(Enhanced Host Controller Interface)是USB 2.0规范中定义的一种主机控制器接口标准。EHCI的设计允许...
在本课程"跟开涛学SpringMVC(4.6)Controller接...在学习过程中,结合"跟开涛学SpringMVC(4.6)Controller接口控制器详解(6)Java开发Java经验技巧共10页.pdf"这份资料,你将能深入理解并掌握SpringMVC的核心概念。
除了设置控制器工厂之外,`ControllerBuilder`还可以用于注册自定义的控制器筛选器(Controller Filters)以及其他相关组件。这使得开发人员能够更加灵活地控制控制器的行为。 #### 实例演示:如何提升命名空间的...
本篇将详细介绍《NFC Controller Interface (NCI) Technical Specification》的核心内容和技术要点,旨在帮助读者深入理解NFC控制器接口的设计与实现。 #### 二、NFC控制器接口(NCI)概述 NFC控制器接口(NCI)是...
《跟开涛学SpringMVC(4.3)Controller接口控制器详解》是针对Java开发者的一份深入学习资料,主要探讨了SpringMVC框架中的Controller组件。SpringMVC是Spring框架的一部分,专门用于处理Web应用程序的请求和响应。...
3. Controller(控制器)作为中介,处理用户输入,将操作转换为对模型的相应调用,并可能影响视图的更新。 在MVC模式中,面向接口编程体现在以下方面: - 视图和模型之间并不直接了解彼此的具体实现,而是通过接口...
### CAN接口的电平详解 #### 一、概述 CAN(Controller Area Network)总线作为一种广泛应用的现场总线协议,其通信方式与电平转换机制是理解CAN接口工作原理的关键。本文将详细介绍CAN接口中显性电平和隐性电平的...
### DDR3控制器详解 DDR3(Double Data Rate 3)是一种高速率的动态随机存取存储器技术,广泛应用于计算机系统、服务器以及其他高性能计算设备中。由于DDR3内存具有较高的带宽需求,其控制器的设计变得至关重要。在...
**ACHI(Advanced Host Controller Interface)**是一种标准化的硬件接口规范,用于连接计算机主板上的串行ATA(SATA)控制器与存储设备。它由Intel在2004年提出,并被广泛采纳为下一代硬盘接口的标准之一。ACHI的...
** SATA 3.0 控制器代码详解 ** SATA(Serial Advanced Technology Attachment)是一种高速接口标准,用于连接计算机系统和存储设备,如硬盘、光驱等。SATA 3.0是SATA规范的第三版,其最大传输速度可达6 Gbps,理论...
《DSI控制器在FPGA/CPLD中的应用详解》 DSI(Display Serial Interface)控制器是一种广泛应用于现代显示系统的接口技术,它通过串行方式传输数据,为显示器提供高效、低功耗的数据通信。在本设计示例"dsi_...
3. **Controller接口控制器** Controller是SpringMVC中处理用户请求的核心组件。控制器通常通过实现Controller接口或者使用注解@Controller来标识。它们负责接收请求参数,调用业务逻辑,最后将结果返回给视图层。 ...