SpringMvc源码分析


第九章 Spring Aop 源码解析(二)



文章目录

  • SpringMvc源码分析
  • 前言
  • 一、监听器、过滤器和拦截器对比
  • 二、自定义拦截器
  • 三、multipart 文件上传
  • 四、springMvc 异常处理
  • 五、springMvc 重定向参数传递



前言


这一章来简单的介绍一下springMvc的高级应用


一、监听器、过滤器和拦截器对比

过滤器(Filter):对Request请求起到过滤的作用,作用在Servlet之前,如果配置为 /* 可以对所有的资源访问(servlet、js/css静态资源等)进行过滤处理

监听器(Listener):实现了javax.servlet.ServletContextListener接口的服务器端组件,它随Web应用的启动而启动,只初始化⼀次,然后会⼀直运行监视,随Web应用的停止而销毁
● 作用⼀:做⼀些初始化工作,web应用中spring容器启动ContextLoaderListener
● 作用⼆:监听web中的特定事件,比如HttpSession,ServletRequest的创建和销毁;变量的创建、销毁和修改等。可以在某些动作前后增加处理,实现监控,比如统计在线人数,利用HttpSessionLisener等。

拦截器(Interceptor):是SpringMVC、Struts等表现层框架自己的,不会拦截jsp/html/css/image的访问等,只会拦截访问的控制器⽅法(Handler)。

从配置的⻆度也能够总结发现:serlvet、filter、listener是配置在web.xml中的,而interceptor是配置在表现层框架⾃⼰的配置⽂件中的
● 在Handler业务逻辑执行之前拦截⼀次
● 在Handler逻辑执行完毕但未跳转页面之前拦截⼀次
● 在跳转页面之后拦截⼀次

梳理spring源码 spring5源码分析_mvc

二、自定义拦截器

自定义拦截器需要实现HandlerInterceptor接口:

public class MyIntercepter01 implements HandlerInterceptor {
    /**
     * 会在handler方法业务逻辑执行之前执行
     * 往往在这里完成权限校验工作
     * @param request
     * @param response
     * @param handler
     * @return  返回值boolean代表是否放行,true代表放行,false代表中止
     * @throws Exception
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("MyIntercepter01 preHandle......");
        return true;
    }


    /**
     * 会在handler方法业务逻辑执行之后尚未跳转页面时执行
     * @param request
     * @param response
     * @param handler
     * @param modelAndView  封装了视图和数据,此时尚未跳转页面呢,你可以在这里针对返回的数据和视图信息进行修改
     * @throws Exception
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("MyIntercepter01 postHandle......");
    }

    /**
     * 页面已经跳转渲染完毕之后执行
     * @param request
     * @param response
     * @param handler
     * @param ex  可以在这里捕获异常
     * @throws Exception
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("MyIntercepter01 afterCompletion......");
    }
}

需要在配置文件中注册一下:

<mvc:interceptors>
       <!--拦截所有handler-->
       <!--<bean class="com.grs.study.interceptor.MyIntercepter01"/>-->
       
       <mvc:interceptor>
           <!--配置当前拦截器的url拦截规则,**代表当前目录下及其子目录下的所有url-->
           <mvc:mapping path="/**"/>
           <!--exclude-mapping可以在mapping的基础上排除一些url拦截-->
           <mvc:exclude-mapping path="/demo/**"/>
           <bean class="com.grs.study.interceptor.MyIntercepter01"/>
       </mvc:interceptor>


       <mvc:interceptor>
           <mvc:mapping path="/**"/>
           <bean class="com.grs.study.interceptor.MyIntercepter02"/>
       </mvc:interceptor>
       
   </mvc:interceptors>

需要注意的是如果配置多个拦截器,存在执行顺序的问题:

梳理spring源码 spring5源码分析_mvc_02

三、multipart 文件上传

页面:

<div>
        <h2>multipart 文件上传</h2>
        <fieldset>
            <form method="post" enctype="multipart/form-data" action="/demo/upload">
                <input type="file" name="uploadFile"/>
                <input type="submit" value="上传"/>
            </form>
        </fieldset>
    </div>

controller:

/**
     * 文件上传
     * @return
     */
    @RequestMapping(value = "/upload")
    public ModelAndView upload(MultipartFile uploadFile,HttpSession session) throws IOException {

        // 处理上传文件
        // 重命名,原名123.jpg ,获取后缀
        String originalFilename = uploadFile.getOriginalFilename();// 原始名称
        // 扩展名  jpg
        String ext = originalFilename.substring(originalFilename.lastIndexOf(".") + 1, originalFilename.length());
        String newName = UUID.randomUUID().toString() + "." + ext;

        // 存储,要存储到指定的文件夹,/uploads/yyyy-MM-dd,考虑文件过多的情况按照日期,生成一个子文件夹
        String realPath = session.getServletContext().getRealPath("/uploads");
        String datePath = new SimpleDateFormat("yyyy-MM-dd").format(new Date());
        File folder = new File(realPath + "/" + datePath);

        if(!folder.exists()) {
            folder.mkdirs();
        }


        // 存储文件到目录
        uploadFile.transferTo(new File(folder,newName));


        // TODO 文件磁盘路径要更新到数据库字段

        Date date = new Date();
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.addObject("date",date);
        modelAndView.setViewName("success");
        return modelAndView;
    }

配置上传解析器(spring.xml)中:

<!--多元素解析器
        id固定为multipartResolver
    -->
    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!--设置上传文件大小上限,单位是字节,-1代表没有限制也是默认的-->
        <property name="maxUploadSize" value="5000000"/>
    </bean>

四、springMvc 异常处理

全局异常捕获:

// 可以让我们优雅的捕获所有Controller对象handler方法抛出的异常
@ControllerAdvice
public class GlobalExceptionResolver {
    @ExceptionHandler(ArithmeticException.class)
    public ModelAndView handleException(ArithmeticException exception, HttpServletResponse response) {
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.addObject("msg",exception.getMessage());
        modelAndView.setViewName("error");
        return modelAndView;
    }
}

五、springMvc 重定向参数传递

转发:A 找 B 借钱400,B没有钱但是悄悄的找到C借了400块钱给A
— url不会变,参数也不会丢失,一个请求
重定向:A 找 B 借钱400,B 说我没有钱,你找别人借去,那么A 又带着400块的借钱需求找到C
— url会变,参数会丢失需要重新携带参数,两个请求

@RequestMapping("/handleRedirect")
 public String handleRedirect(String name,RedirectAttributes redirectAttributes) {
     //return "redirect:handle01?name=" + name;  // 拼接参数安全性、参数长度都有局限
     // addFlashAttribute方法设置了一个flash类型属性,该属性会被暂存到session中,在跳转到页面之后该属性销毁
     redirectAttributes.addFlashAttribute("name",name);
     return "redirect:handle01";

 }