欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

SpringMVC学习笔记(四)

程序员文章站 2022-07-15 11:41:43
...

1、AbstractWizardFormController

       AbstractWizardFormController 能够实现向导式的页面。如果用户需要填写的表单内容很多,就有必要将其拆为几个页面,使用户能通过“上一步”和“下一步”按钮方便地在向导页面间导航,例 如,设计一个在线调查的向导,就可以方便地引导用户一步一步完成调查表单的填写。

我们以注册新用户为例,RegisterController需要用户填写基本资料、联系方式和详细地址,由于表单内容较多,我们让用户分3个页面分步完成注册。

我们无须处理“下一步”和“上一步”按钮,Spring会自动显示正确的页面,我们只需要处理最后用户单击“完成”按钮提交的整个表单对象。

/**

 * 用户注册向导

 * @spring.bean name="/register.do"

 * @spring.property name="commandClass" value="example.chapter7.User"

 * @spring.property name="pages" list="registerStep0,registerStep1, registerStep2"

 */

public class RegisterController extends AbstractWizardFormController {

    private UserService userService;

    /**

     * @spring.property ref="userService"

     */

    public void setUserService(UserService userService) {

        this.userService = userService;

    }

    // 当用户单击"_finish"按钮时,触发processFinish()方法:

    protected ModelAndView processFinish(HttpServletRequest request,

            HttpServletResponse response, Object command, BindException errors)

            throws Exception {

        User user = (User)command;

        userService.register(user);

        Map model = new HashMap();

        model.put("username", user.getUsername());

        return new ModelAndView("registerSuccess", model);

    }

}
 

那 么,Spring是如何知道下一个或上一个需要显示的页面呢?

除了指定command Class为User对象外,我们还需要将几个View的逻辑名称注入到RegisterController的pages属性中,注意到 AbstractWizardController的pages是从下标0开始计数的,因此,我们将注册的3个页面依次命名为 registerStep0.jsp、registerStep1.jsp和registerStep2.jsp。

除了指定pages属性外,我们还需要按照一定的规则来编写JSP页面,才能告诉Spring如何显示下一页或上一页。在表单的提交按钮上,必须以_target+索引命名按钮 ,例如:

 

<input type="submit" name="_target1" value="下一步">将前进到索引为1的页面,即registerStep1.jsp。

<input type="submit" name="_target0" value="上一步">将返回到索引为0的页面,即registerStep0.jsp。

最后一个“完成”按钮必须以“_finish”命名。

<input type="submit" name="_finish" value="完成">
 

当用户单击“完成”按钮后,Spring将调用processFinish()方法处理表单。

如果需要验证表单,在AbstractWizardController中,就无法使用Validator来进行验证,因为用户在每个页面仅填写了部分内 容,直到用户单击“完成”按钮时,整个表单对象才被填充完毕,因此,在任何一个页面中验证Command都将失败,为此,验证必须在 AbstractWizardController的validatePage()方法中进行,Spring将传入page参数,我们就根据这个参数对 command对象进行部分验证。

// 每当用户单击"_target?"准备前进到下一步时,触发validatePage()来验证当前页:

protected void validatePage(Object command, Errors errors, int page) {

    User user = (User)command;

    if(page==0) {

        // 验证username,password:

        if(!user.getUsername().matches("[a-zA-Z0-9]{3,20}"))

            errors.rejectValue("username", "error.username", "用户名不符合要求");

        if(userService.isExist(user.getUsername()))

            errors.rejectValue("username", "error.username", "用户名已存在");

        if(user.getPassword()==null || user.getPassword().length()<6)

            errors.rejectValue("password", "error.password", "口令至少为6个字符");

    }

    else if(page==1) {

        // 验证email,blog,website:

        if(user.getEmail()==null)

            errors.rejectValue("email", "error.email.empty", "电子邮件不能为空");

        else if(!user.getEmail().matches("[a-zA-Z0-9\\_\\-]+\\@[a-zA-Z0-9\\_ \\-]+[\\.[a-zA-Z0-9\\_\\-]+]{1,2}"))

            errors.rejectValue("email", "error.email", "电子邮件地址无效");

        if(user.getBlog()==null || user.getBlog().trim().equals(""))

            errors.rejectValue("blog", "error.blog", "博客地址不能为空");

        if(user.getWebsite()==null || user.getWebsite().trim().equals(""))

            errors.rejectValue("website", "error.website", "网址地址不能为空");

    }

    else if(page==2) {

        // 验证province,city,zip:略过

    }

}

 若验证未通过,则将停留在当前页,并可以通过<form:errors>显示相应的错误信息,待用户更正后,才可以继续前进到下一页。

也许注意到了,第一个页面有两个口令框,其中,第二个口令框名称为password2,在User对象中并没有对应的属性,Spring不会自动绑定它。 那么,如何验证用户两次输入的口令是否一致呢?我们一般不愿意更改User对象,因为User对象很可能对应数据库中的某个表,而数据库表不会存储同一用 户的两份相同的口令。此时,可以通过 JavaScript来验证,既方便,又能避免修改User对象。 因此,在Web应用程序的设计中,不要仅仅拘泥于JavaEE框架,对于 JavaScript、AJAX等技术也要充分利用。


2、重定向URL

 

重定向URL会使服务器向客户端发送一个 Redirect响应,并包含一个目标URL。客户端接收到Redirect响应后,会立刻重新请求新的URL,这一点和Forward不同。前者使客户 端发送了两次独立的HTTP请求,而后者请求是在服务器内部处理的,客户端并不知道服务器端对Request是否做了Forward处理。

重定向功能的主要用途是为了在服务器端修改了某 一资源的URL后,原有客户仍可以继续通过原来的URL访问该资源。由于重定向会使客户端发送两次请求,所以降低了网络效率,并且不便于用户在浏览器中单 击“后退”按钮返回上一个页面。对于Web应用程序而言,决不能大量使用重定向功能。

在Controller中实现 Redirect也非常容易。最简单的方法是直接调用HttpServlet Response对象的sendRedirect()方法,然后返回null。一旦返回的ModelAndView为null,Spring就认为 Controller自己已经完成了请求处理,不再按照常规的MVC流程继续处理请求。

例如,对于用户注销登录的操作,在清理了Session的内容后,就可以将用户重定向到登录页面。LogoutController代码如下。

 

/**

 * @spring.bean name="/logout.do"

 */

public class LogoutController extends AbstractController {

    protected ModelAndView handleRequestInternal(HttpServletRequest request, HttpServletResponse response) throws Exception {

        request.getSession().removeAttribute("USERNAME");

        response.sendRedirect("login.do");

        return null;

    }

}
 

另一种 实现重定向的方法不用直接调用HttpServletResponse对象的sendRedirect()方法,而是返回一个带有 “redirect:”前缀的View,这样,ViewResolver就知道这是一个重定向操作,于是不再渲染视图,而是直接向客户端发送 Redirect响应。

return new ModelAndView("redirect:login.do");

Spring还提供了一个RedirectView对象,也可以实现重定向操作,不过使用RedirectView使Controller和View的耦合稍微紧密了一点,推荐的方法是使用“redirect:”前缀。

使用重定向要注意的一点是,重定向的资源不可位于/WEB-INF/目录下,因为用户无法通过URL直接访问位于/WEB-INF/目录下的资源,而使用MVC流程通过forward调用/WEB-INF/目录下的资源是允许的。

 

3、处理异常

       如果Controller在处理用户请求时发生了异常,自己捕获异常并跳转到出错页面会使核心逻辑混乱。Spring的MVC框架提供了一个HandlerExceptionResolver,为所有的Controller抛出的异常提供一个统一的入口。

/**

 * @spring.bean id="handlerExceptionResolver"

 */

public class MyHandlerExceptionResolver implements HandlerExceptionResolver {

    private Log log = LogFactory.getLog(getClass());

    public ModelAndView resolveException(HttpServletRequest request,

            HttpServletResponse response, Object handler, Exception ex) {

        log.warn("Handle exception: " + ex.getClass().getName());

        if(ex instanceof NeedLoginException)

            return new ModelAndView("redirect:login.do");

        Map model = new HashMap();

        model.put("ex", ex.getClass().getSimpleName());

        model.put("message", ex.getMessage());

        return new ModelAndView("error", model);

    }

}
 

MyHandlerExceptionResolver 根据Exception类型判断如何处理异常,如果是NeedLoginException,说明系统要求用户登录,这时直接将用户导向到登录页面;对于 其他类型的异常,则直接将异常的错误信息显示给用户,注意返回的视图名称为“error”,实际的视图文件即为“/error.jsp”。

使用 HandlerExceptionResolver可以避免在应用程序的每一个Controller中都去处理异常,将异常统一放到 HandlerExceptionResolver中可以极大地简化异常处理逻辑,也便于在一个统一的地方记录异常日志。对于无法处理的异常,可以给用户 显示一个友好的出错页面。

 

4、拦截请求

       我们已使用 Filter可以拦截用户请求,并实现相应的处理。Spring的MVC框架也提供了一个拦截器链,可以由多个HandlerInterceptor构 成,允许在Controller处理用户请求的前后有机会处理请求。和Filter相比,HandlerInterceptor是在Spring的IoC 容器中配置的,可以注入任意的组件,而Filter定义在Spring容器之外,因此,注入IoC组件比较困难,或者难以得到一个优雅的设计。

HandlerInterceptor接口定义了以下3个方法。

boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)

void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
 

(1)preHandle()方法在Controller执行前调用,其返回值指定了是否应当继续处理请求。若返回false,Spring MVC框架将不再继续调用下一个拦截器,也不会将请求交给Controller处理,整个请求处理将到此结束。

(2)postHandler()方法在Controller执行完毕后调用,此时Controller仅返回了ModelAndView对象,还没有对视图进行渲染,在这个方法中有机会对ModelAndView进行修改。

(3)afterCompletion()方法在整个请求全部完成后调用,通过判断参数ex是否为null就可以判断是否产生了异常。

通过HandlerInterceptor,就有机会在一个请求执行的3个阶段对其进行拦截。例如,为了统计Web应用程序的性能,我们设计了一个性能拦截器,将每个用户请求的处理时间记录下来。PerformanceHandlerInterceptor实现如下。

/**

 * @spring.bean id="performanceHandler"

 */

public class PerformanceHandlerInterceptor implements HandlerInterceptor {

    private final Log log = LogFactory.getLog(PerformanceHandlerInterceptor. class);

    private static final String START_TIME = "PERF_START";

    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        request.setAttribute(START_TIME, System.currentTimeMillis());

        return true;

    }

    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {

        // 不需要处理postHandler, 保留空方法即可

    }

    public void afterCompletion(HttpServletRequest request, HttpServlet Response response, Object handler, Exception ex) throws Exception {

        Long startTime = (Long)request.getAttribute(START_TIME);

        if(startTime!=null) {

            long last = System.currentTimeMillis() - startTime.longValue();

            String uri = request.getRequestURI();

            String query = request.getQueryString();

            if(query!=null)

                uri = uri + '?' + query;

            log.info("URL: " + uri);

            log.info("Execute: " + last + "ms.");

        }

    }

}
 

由于我 们必须保证PerformanceHandlerInterceptor是线程安全的,因此,绝不可将起始时间记录在 PerformanceHandlerInterceptord 的成员变量中。由于每个请求都对应一个独立的HttpServletRequest实例,因此,将起始时间放入HttpServletRequest实例 中就保证了线程安全。

然后,将其添加到handlerMapping中的interceptor列表中。

<bean id="handlerMapping" class="org.springframework.web.servlet.handler. BeanNameUrlHandlerMapping">

    <property name="interceptors">

        <list>

            <ref bean="performanceHandler" />

        </list>

    </property>

</bean>
 

运行应用程序,在浏览器中请求/login.do,查看控制台输出如下。

[2006/11/20 22:33:41.857] URL: /login.do

[2006/11/20 22:33:41.857] Execute: 3781ms.

[2006/11/20 22:33:45.325] URL: /login.do

[2006/11/20 22:33:45.325] Execute: 0ms.

可以看到PerformanceHandlerInterceptor记录的处理时间。首次执行/login.do请求时,耗时3秒多,这是因为服务器需要编译JSP文件,随后刷新页面,由于可以跳过JSP的编译步骤,/login.do请求在1ms内就完成了。

 

5、处理文件上传

文件上传是Web应用程序中常见的功能。本质上,浏览器在向服务器发送文件时,其HTTP请求必须以multipart/form-data的形式发送,该规范定义在RFC 2388(http://www.ietf.org/rfc/rfc2388.txt)中,可以实现一次上传一个或多个文件。不过,JavaEE的Web 规范并没有内置处理multipart请求的功能,因此,要实现文件上传,就必须借助于第三方组件,或者自己手动编码解析 HttpServletRequest。

Apache Commons FileUpload(http://jakarta.apache.org/commons/fileupload)组件和COS FileUpload(http://www.servlets.com/cos)组件都是常见的处理文件上传的组件,Spring很好地对这两种组件进 行了封装。在Spring中处理文件上传时,根本无须与这两个组件的API打交道,只需用到Spring提供的 MultipartHttpServletRequest对象,就可以轻松实现文件上传的功能。

默认地,Spring不会处理文件上传,即所有的以multipart/form-data形式发送的请求都不被处理,如果要处理Multipart请求,需要在Spring的XML配置文件中申明一个MultipartResolver。

<bean id="multipartResolver" class="org.springframework.web.multipart. commons.CommonsMultipartResolver">

    <!-- 最大允许上传文件大小:1M -->

    <property name="maxUploadSize" value="1048576" />

</bean>

 maxUploadSize属性指定了最大所能上传的文件大小,若超出了最大范围,Spring将会直接抛出异常。

如果一个请求不是Multipart请求,它就会按照正常的流程处理;

如果一个请求是Multipart请求,Spring就会自动调用MultipartResolver,然后将 HttpServletRequest请求变为MultipartHttpServletRequest请求,开发者只需要处理 MultipartHttpServletRequest对象就可以了。

如何得知一个请求是否是MultipartHttpServletRequest类型呢?通过instanceof操作就能非常简单地判断出来。 我们在UploadController中实现文件上传的代码如下。

public class UploadController implements Controller {

    private Log log = LogFactory.getLog(getClass());

    public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {

        // 判断request是不是multipart请求:

        if(request instanceof MultipartHttpServletRequest) {

            MultipartHttpServletRequest multipart = (MultipartHttpServlet Request)request;

            MultipartFile file = multipart.getFile("file");

            if(file==null || file.isEmpty()) {

                // 文件不存在:

                response.sendError(HttpServletResponse.SC_BAD_REQUEST);

                return null;

            }

            String filename = file.getOriginalFilename();

            log.info("Upload file name: " + filename);

            // 获取文件扩展名:

            String ext = "";

            int pos;

            if((pos = filename.lastIndexOf('.'))!=(-1)) {

                ext = URLEncoder.encode(filename.substring(pos).trim(), "UTF-8");

            }

            InputStream input = null;

            OutputStream output = null;

            // 确定服务器端写入文件的文件名:

            String uploadFile = request.getSession()

                    .getServletContext()

                    .getRealPath("/upload" + System.currentTimeMillis() + ext);

            try {

                // 获得上传文件的输入流:

                input = file.getInputStream();

                // 写入到服务器的本地文件:

                output = new BufferedOutputStream(new FileOutputStream(uploadFile));

                byte[] buffer = new byte[1024];

                int n;

                while((n=input.read(buffer))!=(-1)) {

                    output.write(buffer, 0, n);

                }

            }

            finally {

                // 必须在finally中关闭输入/输出流:

                if(input!=null) {

                    try {

                        input.close();

                    }

                    catch(IOException ioe) {}

                }

                if(output!=null) {

                    try {

                        output.close();

                    }

                    catch(IOException ioe) {}

                }

            }

            // 告诉浏览器文件上传成功:

            Writer writer = response.getWriter();

            writer.write("File uploaded successfully!");

            writer.flush();

        }

        else {

            // 非multipart/form-data请求,发送一个错误:

            response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);

        }

        return null;

    }

}
 

仔细查看上面的代码,可能会发现,我们根本没有调用Commons FileUpload或COS FileUpload组件的API,Spring已经完全为我们封装好了。那么,Spring如何确定使用Commons FileUpload还是使用COS FileUpload呢?答案是发现哪个就用哪个。

如果在/WEB-INF/lib目录下放置Commons FileUpload的jar包,Spring就会自动使用Commons FileUpload,COS FileUpload也是如此,这样带来的好处是完全屏蔽了底层组件的API,如果需要替换底层组件,只需要替换相应的jar包,甚至连XML配置文件都不用改动。

在 WebUpload工程中,我们使用的是Commons FileUpload,只需将commons- fileupload.jar和commons-io.jar放到/WEB-INF/lib目录下,剩下的事情就由Spring处理了。使用任何文本编辑器编写一个最简单的上传文件的index.html页面。

<html><head>

<meta http-equiv="Content-Type" content="text/html; charset=utf-8">

<title>Upload File</title>

</head>

<body>

<form action="upload.do" method="post" enctype="multipart/form-data" name="form1">

<p>请选择需要上传的文件:<input type="file" name="file"></p>

<p><input type="submit" name="Submit" value="上传"></p>

</form>

</body>

</html>
 

配置好DispatcherServlet后,运行这个Web应用程序,打开index.html,选择待上传的文件。文件上传成功后,就可以在服务器的Web应用的根目录下找到已上传的文件。


对于非file类型的表单字段,仍可以调用MultipartHttpServletRequest的getParameter()方法获得相应的字段值,因为MultipartHttpServletRequest也实现了HttpServletRequest接口。

也可以在SimpleFormController中将表单中上传的文件绑定到byte[]类型的属性中,不过,如果上传文件较大,则将消耗较大的服务器内存,因此,采用何种解决方案需要视情况而定。