学习SpringMVC系列教程三Controller接口控制器详解2.docx
《学习SpringMVC系列教程三Controller接口控制器详解2.docx》由会员分享,可在线阅读,更多相关《学习SpringMVC系列教程三Controller接口控制器详解2.docx(10页珍藏版)》请在冰豆网上搜索。
学习SpringMVC系列教程三Controller接口控制器详解2
学习SpringMVC系列教程(三)Controller接口控制器详解
(2)
4.5、ServletForwardingController
将接收到的请求转发到一个命名的servlet,具体示例如下:
JavaCode复制内容到剪贴板
packagecn.javass.chapter4.web.servlet;
publicclassForwardingServletextendsHttpServlet{
@Override
protectedvoiddoGet(HttpServletRequestreq,HttpServletResponseresp)
throwsServletException,IOException{
resp.getWriter().write("ControllerforwardtoServlet");
}
}
XML/HTMLCode复制内容到剪贴板
<servlet>
<servlet-name>forwarding</servlet-name>
<servlet-class>cn.javass.chapter4.web.servlet.ForwardingServlet</servlet-class>
</servlet>
XML/HTMLCode复制内容到剪贴板
<!
—在chapter4-servlet.xml配置处理器-->
<beanname="/forwardToServlet"
class="org.springframework.web.servlet.mvc.ServletForwardingController">
<propertyname="servletName"value="forwarding"></property>
</bean>
当我们请求/forwardToServlet时,会被转发到名字为“forwarding”的servlet处理,该sevlet的servlet-mapping标签配置是可选的。
4.6、BaseCommandController
命令控制器通用基类,提供了以下功能支持:
1、数据绑定:
请求参数绑定到一个commandobject(命令对象,非GoF里的命令设计模式),这里的命令对象是指绑定请求参数的任何POJO对象;
commandClass:
表示命令对象实现类,如UserModel;
commandName:
表示放入请求的命令对象名字(默认command),request.setAttribute(commandName,commandObject);
2、验证功能:
提供Validator注册功能,注册的验证器会验证命令对象属性数据是否合法;
validators:
通过该属性注入验证器,验证器用来验证命令对象属性是否合法;
该抽象类没有没有提供流程功能,只是提供了一些公共的功能,实际使用时需要使用它的子类。
4.7、AbstractCommandController
命令控制器之一,可以实现该控制器来创建命令控制器,该控制器能把自动封装请求参数到一个命令对象,而且提供了验证功能。
1、创建命令类(就是普通的JavaBean类/POJO)
JavaCode复制内容到剪贴板
packagecn.javass.chapter4.model;
publicclassUserModel{
privateStringusername;
privateStringpassword;
//省略setter/getter
}
2、实现控制器
JavaCode复制内容到剪贴板
packagecn.javass.chapter4.web.controller;
//省略import
publicclassMyAbstractCommandControllerextendsAbstractCommandController{
publicMyAbstractCommandController(){
//设置命令对象实现类
setCommandClass(UserModel.class);
}
@Override
protectedModelAndViewhandle(HttpServletRequestreq,HttpServletResponseresp,Objectcommand,BindExceptionerrors)throwsException{
//将命令对象转换为实际类型
UserModeluser=(UserModel)command;
ModelAndViewmv=newModelAndView();
mv.setViewName("abstractCommand");
mv.addObject("user",user);
returnmv;
}
}
XML/HTMLCode复制内容到剪贴板
<!
—在chapter4-servlet.xml配置处理器-->
<beanname="/abstractCommand"
class="cn.javass.chapter4.web.controller.MyAbstractCommandController">
<!
--也可以通过依赖注入注入命令实现类-->
<!
--propertyname="commandClass"value="cn.javass.chapter4.model.UserModel"/-->
</bean>
JavaCode复制内容到剪贴板
<!
—WEB-INF/jsp/abstractCommand.jsp视图下的主要内容-->
${user.username}-${user.password}
当我们在浏览器中输入“http:
//localhost:
9080/springmvc-chapter4/abstractCommand?
username=123&password=123”,会自动将请求参数username和password绑定到命令对象;绑定时按照JavaBean命名规范绑定;
4.8、AbstractFormController
用于支持带步骤的表单提交的命令控制器基类,使用该控制器可以完成:
1、定义表单处理(表单的渲染),并从控制器获取命令对象构建表单;
2、提交表单处理,当用户提交表单内容后,AbstractFormController可以将用户请求的数据绑定到命令对象,并可以验证表单内容、对命令对象进行处理。
JavaCode复制内容到剪贴板
@Override
rotectedModelAndViewhandleRequestInternal(HttpServletRequestrequest,HttpServletResponseresponse)
throwsException{
//1、是否是表单提交?
该方法实现为("POST".equals(request.getMethod())),即POST表示表单提交
if(isFormSubmission(request)){
try{
Objectcommand=getCommand(request);
ServletRequestDataBinderbinder=bindAndValidate(request,command);
BindExceptionerrors=newBindException(binder.getBindingResult());
//表单提交应该放到该方法实现
returnprocessFormSubmission(request,response,command,errors);
}
catch(HttpSessionRequiredExceptionex){
//省略部分代码
returnhandleInvalidSubmit(request,response);
}
}
else{
//2、表示是表单展示,该方法又转调showForm方法,因此我们需要覆盖showForm来完成表单展示
returnshowNewForm(request,response);
}
bindOnNewForm:
是否在进行表单展示时绑定请求参数到表单对象,默认false,不绑定;
sessionForm:
session表单模式,如果开启(true)则会将表单对象放置到session中,从而可以跨越多次请求保证数据不丢失(多步骤表单常使用该方式,详解AbstractWizardFormController),默认false;
ObjectformBackingObject(HttpServletRequestrequest):
提供给表单展示时使用的表单对象(formobject表单要展示的默认数据),默认通过commandName暴露到请求给展示表单;
MapreferenceData(HttpServletRequestrequest,Objectcommand,Errorserrors):
展示表单时需要的一些引用数据(比如用户注册,可能需要选择工作地点,这些数据可以通过该方法提供),如:
JavaCode复制内容到剪贴板
protectedMapreferenceData(HttpServletRequestrequest)throwsException{
Mapmodel=newHashMap();
model.put("cityList",cityList);
returnmodel;
}
这样就可以在表单展示页面获取cityList数据。
SimpleFormController继承该类,而且提供了更简单的表单流程控制。
4.9、SimpleFormController
提供了更好的两步表单支持:
1、准备要展示的数据,并到表单展示页面;
2、提交数据数据进行处理。
第一步,展示:
第二步,提交表单:
接下来咱们写一个用户注册的例子学习一下:
(1、控制器
JavaCode复制内容到剪贴板
packagecn.javass.chapter4.web.controller;
//省略import
publicclassRegisterSimpleFormControllerextendsSimpleFormController{
publicRegisterSimpleFormController(){
setCommandClass(UserModel.class);//设置命令对象实现类
setCommandName("user");//设置命令对象的名字
}
//formobject表单对象,提供展示表单时的表单数据(使用commandName放入请求)
protectedObjectformBackingObject(HttpServletRequestrequest)throwsException{
UserModeluser=newUserModel();
user.setUsername("请输入用户名");
returnuser;
}
//提供展示表单时需要的一些其他数据
protectedMapreferenceData(HttpServletRequestrequest)throwsException{
Mapmap=newHashMap();
map.put("cityList",Arrays.asList("山东","北京","上海"));
returnmap;
}
protectedvoiddoSubmitAction(Objectcommand)throwsException{
UserModeluser=(UserModel)command;
//TODO调用业务对象处理
System.out.println(user);
}
}
setCommandClass和setCommandName:
分别设置了命令对象的实现类和名字;
formBackingObject和referenceData:
提供了表单展示需要的视图;
doSubmitAction:
用于执行表单提交动作,由onSubmit方法调用,如果不需要请求/响应对象或进行数据验证,可以直接使用doSubmitAction方法进行功能处理。
(2、spring配置(chapter4-servlet.xml)
XML/HTMLCode复制内容到剪贴板
<beanname="/simpleForm"
class="cn.javass.chapter4.web.controller.RegisterSimpleFormController">
<propertyname="formView"value="register"/>
<propertyname="successView"value="redirect:
/success"/>
</bean>
<beanname="/success"class="cn.javass.chapter4.web.controller.SuccessController"/>
formView:
表示展示表单时显示的页面;
successView:
表示处理成功时显示的页面;“redirect:
/success”表示成功处理后重定向到/success控制器;防止表单重复提交;
“/success”bean的作用是显示成功页面,此处就不列举了。
(3、视图页面
XML/HTMLCode复制内容到剪贴板
<!
--register.jsp注册展示页面-->
<formmethod="post">
username:
<inputtype="text"name="username"value="${user.username}"><br/>
password:
<inputtype="password"name="username"><br/>
city:
<select>
<c:
forEachitems="${cityList}"var="city">
<option>${city}</option>
</c:
forEach>
</select><br/>
<inputtype="submit"value="注册"/>
</form>
此处可以使用${user.username}获取到formBackingObject设置的表单对象、使用${cityList}获取referenceData设置的表单支持数据;
到此一个简单的两步表单到此结束,但这个表单有重复提交表单的问题,而且表单对象到页面的绑定是通过手工绑定的,后边我们会学习spring标签库(提供自动绑定表单对象到页面)。
4.10、CancellableFormController
一个可取消的表单控制器,继承SimpleFormController,额外提供取消表单功能。
1、表单展示:
和SimpleFormController一样;
2、表单取消:
和SimpleFormController一样;
3、表单成功提交:
取消功能处理方法为:
onCancel(Objectcommand),而且默认返回cancelView属性指定的逻辑视图名。
那如何判断是取消呢?
如果请求中有参数名为“_cancel”的参数,则表示表单取消。
也可以通过cancelParamKey来修改参数名(如“_cancel.x”等)。
示例:
(1、控制器
复制RegisterSimpleFormController一份命名为CanCancelRegisterSimpleFormController,添加取消功能处理方法实现:
JavaCode复制内容到剪贴板
@Override
protectedModelAndViewonCancel(Objectcommand)throwsException{
UserModeluser=(UserModel)command;
//TODO调用业务对象处理
System.out.println(user);
returnsuper.onCancel(command);
}
onCancel:
在该功能方法内实现取消逻辑,父类的onCancel方法默认返回cancelView属性指定的逻辑视图名。
(2、spring配置(chapter4-servlet.xml)
XML/HTMLCode复制内容到剪贴板
<beanname="/canCancelForm"
class="cn.javass.chapter4.web.controller.CanCancelRegisterSimpleFormController">
<propertyname="formView"value="register"/>
<propertyname="successView"value="redirect:
/success"/>
<propertyname="cancelView"value="redirect:
/cancel"/>
</bean>
<beanname="/cancel"class="cn.javass.chapter4.web.controller.CancelController"/>
cancelParamKey:
用于判断是否是取消的请求参数名,默认是_cancel,即如果请求参数数据中含有名字_cancel则表示是取消,将调用onCancel功能处理方法;
cancelView:
表示取消时时显示的页面;“redirect:
/cancel”表示成功处理后重定向到/cancel控制器;防止表单重复提交;
“/cancel”bean的作用是显示取消页面,此处就不列举了(详见代码)。
(3、视图页面(修改register.jsp)
XML/HTMLCode复制内容到剪贴板
<inputtype="submit"name="_cancel"value="取消"/>
该提交按钮的作用是取消,因为name="_cancel",即请求后会有一个名字为_cancel的参数,因此会执行onCancel功能处理方法。
(4、测试:
在浏览器输入“http:
//localhost:
9080/springmvc-chapter4/canCancelForm”,则首先到展示视图页面,点击“取消按钮”将重定向到“http:
//localhost:
9080/springmvc-chapter4/cancel”,说明取消成功了。
实际项目可能会出现比如一些网站的完善个人资料都是多个页面(即多步),那应该怎么实现呢?
接下来让我们看一下springWebMVC提供的对多步表单的支持类AbstractWizardFormController。