当前位置: 首页 > news >正文

Spring MVC拦截器、文件上传和全局异常处理

目录

  • 1.拦截器
    • 1.1.什么是拦截器?
    • 1.2 拦截器的API
    • 1.3 拦截器的执行顺序
    • 1.5 自定义拦截器
    • 1.5 登录拦截器案例
  • 2.文件上传
    • 2.1 添加依赖
    • 2.2 配置文件上传解析器
    • 2.3 编写控制器
    • 2.4 编写jsp页面
    • 2.5 注意事项
  • 3.全局异常处理器
    • 3.1 异常处理思路
    • 3.2 创建异常处理器
    • 3.3 编写异常类
    • 3.4 配置异常处理器

1.拦截器

1.1.什么是拦截器?

​ Spring MVC中的拦截器(Interceptor)类似于Servlet中的过滤器(Filter),它主要用于拦截用户请求并作相应的处理。例如通过拦截器可以进行权限验证、判断用户是否登录等。

​ 拦截器依赖于web框架,在实现上基于Java的反射机制,属于面向切面编程(AOP)的一种运用。

过滤器与拦截器的区别:拦截器是AOP思想的具体应用。

过滤器:

  • servlet规范中的一部分,任何java web工程都可以使用

  • 在url-pattern中配置了/*之后,可以对所有要访问的资源进行拦截

拦截器:

  • 拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用

  • 拦截器只会拦截访问的控制器方法, 如果访问的是jsp/html/css/image/js是不会进行拦截的

1.2 拦截器的API

定义拦截器需要实现HandlerInterceptor接口,该接口中有三个方法:

  • preHandle()

控制层执行器方法前的拦截器(该方法时在控制层执行器方法前调用,当该方法返回结果为true则继续调用下一个拦截器,如果已经是最后一个拦截器,则调用控制层中的执行器方法;当该方法返回结果为false,则不会继续执行控制层执行器中的方法)

  • postHandle()

控制层方法返回时拦截器(该方法是控制层执行器方法执行之后,由DispatcherServlet在将结果响应给浏览器前调用的方法)

  • afterCompletion()

控制层方法结束后的拦截器(该方法在请求业务处理执行完全结束之后由DispatcherServlet调用执行)

1.3 拦截器的执行顺序

  • 单个拦截器的执行顺序

preHandle()postHandle()afterCompletion()

  • 多个拦截器的执行顺序
  1. 先按顺序执行所有拦截器的 preHandle方法

  2. 如果任何一个拦截器preHandle方法返回false。直接跳出不执行目标方法

    • 如果当前拦截器prehandler返回为true。则执行下一个拦截器的preHandle方法
    • 如果当前拦截器返回为false。直接倒序执行所有已经执行了的拦截器的 afterCompletion方法;
  3. 所有拦截器都返回True。执行目标方法

  4. 倒序执行所有拦截器的postHandle方法。

  5. 前面的步骤有任何异常都会直接倒序触发 afterCompletion

  6. 页面成功渲染完成以后,也会倒序触发 afterCompletion

1.5 自定义拦截器

我们要想自定义拦截器, 要求必须实现:HandlerInterceptor 接口。

  • 创建拦截器

配置web.xml 和 springmvc-servlet.xml 文件

package com.by.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;public class CustomHandlerInterceptor implements HandlerInterceptor {/*** 在控制器方法调用前执行* 返回值为是否中断*      true:表示继续执行(下一个拦截器或处理器)*      false:则会中断后续的所有操作,所以我们需要使用response来继续响应后续请求*/@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object object) throws Exception {System.out.println("HandlerInterceptor preHandle ....");return true;}/*** 在控制器方法调用后,解析视图前调用,我们可以对视图和模型做进一步渲染或修改* 可在modelAndView中加入数据,比如当前时间*/@Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response, Object object, ModelAndView modelAndView) throws Exception {System.out.println("HandlerInterceptor postHandle ....");}/*** 整个请求完成,即视图渲染结束后调用,这个时候可以做些资源清理工作,或日志记录*/@Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response,Object object, Exception e) throws Exception {System.out.println("HandlerInterceptor afterCompletion ....");}
}
  • 配置拦截器
    <!--配置拦截器 --><mvc:interceptors><mvc:interceptor><!-- 匹配的是url路径 --><mvc:mapping path="/**"></mvc:mapping><bean class="com.by.interceptor.CustomHandlerInterceptor"></bean></mvc:interceptor></mvc:interceptors>
  • 测试

  • 编写controller

@Controller
@RequestMapping("/account")
public class AccountController {@RequestMapping("/findAccount12")public String findAccount12(Model model) {model.addAttribute("msg", "欢迎你 springmvc");System.out.println("controller的方法执行了......");return "success";}
}

1.5 登录拦截器案例

  • 创建拦截器
package com.by.interceptor;import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;public class LoginInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object object) throws Exception {String user = (String) request.getSession().getAttribute("user_session");if (user != null) {//已经登录,继续执行System.out.println("获得用户信息:"+user);return true;} else {//未登录,跳转到登录页面response.sendRedirect(request.getContextPath() +"/login.jsp");return false;}}
}
  • 配置拦截器
	<mvc:interceptors><mvc:interceptor><!-- 匹配的是url路径 --><mvc:mapping path="/**"></mvc:mapping><bean class="com.by.interceptor.LoginInterceptor"></bean></mvc:interceptor></mvc:interceptors>
  • 测试
  1. 编写controller
@Controller
@RequestMapping("/account")
public class AccountController {@RequestMapping("/findAccount13")public String findAccount13(Model model) {model.addAttribute("msg", "欢迎你 springmvc");System.out.println("controller的方法执行了......");return "success";}
}
  1. 在index.jsp里面定义超链接
<%session.setAttribute("user_session","User{name:张二狗,age:18}");
%>
<a href="/account/findAccount13">登录拦截器</a>

2.文件上传

2.1 添加依赖

<!--文件上传-->
<dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.3.1</version>
</dependency>

2.2 配置文件上传解析器

<!--配置文件上传解析器-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><property name="maxUploadSize" value="5242880" /><property name="defaultEncoding" value="UTF-8" />
</bean>

2.3 编写控制器

  • 编写controller
@Controller
@RequestMapping("/account")
public class AccountController {@RequestMapping(path="/upload")public String upload(HttpServletRequest request, MultipartFile upload,Model model) throws IOException {System.out.println("springmvc方式的文件上传");//获取要上传的文件目录String path = request.getSession().getServletContext().getRealPath("/uploads");System.out.println("path:"+path);//根据文件上传的目录创建File对象,如果不存在则创建1个File对象File file = new File(path);if(!file.exists()){//创建一个file对象file.mkdirs();}//获取文件上传名称String filename = upload.getOriginalFilename();//完成文件上传upload.transferTo(new File(path,filename));model.addAttribute("msg", "欢迎你 springmvc");return "success";}
}

2.4 编写jsp页面

    <form action="/account/upload" method="post" enctype="multipart/form-data">文件: <input type="file" name="upload"></input><input type="submit" value="提交"></form>

2.5 注意事项

  1. form表单必须加上enctype="multipart/form-data"
  2. method属性取值必须是Post
  3. 提供一个文件选择域<input type="file"/>

3.全局异常处理器

即使对大部分情况都做了预判和异常处理,但程序有时会因为莫名奇妙的原因发生错误,并且错误信息肯定会抛在浏览器页面上,这样用户观感特别不好,所以需要进行全局异常处理。

3.1 异常处理思路

系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理,如下图:

异常处理

3.2 创建异常处理器

@Component
public class CustomExceptionResolver implements HandlerExceptionResolver {@Overridepublic ModelAndView resolveException(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex) {ModelAndView modelAndView = new ModelAndView();modelAndView.addObject("message", ex.getMessage());modelAndView.setViewName("error");return modelAndView;}
}

3.3 编写异常类

@Controller
@RequestMapping("/account")
public class AccountController {@RequestMapping("/findAccount14")public String findAccount14(Model model) {model.addAttribute("msg", "欢迎你 springmvc");//模拟异常信息int i = 10/0;return "success";}
}

3.4 配置异常处理器

<bean class="com.by.exception.CustomExceptionResolver "></bean>

相关文章:

  • Ubuntu下FastDDS的源码编译和简单测试
  • C语言小例程
  • 自动驾驶仿真Carla -ACC功能测试
  • Centos7 Mysql8.3.0 安装地址
  • 数据分析的数学概念
  • GitHub Copilot 登录账号激活,已经在IntellJ IDEA使用
  • 鸿蒙 HarmonyOS NEXT星河版APP应用开发—上篇
  • C#循环语句总结
  • LLM功能应用的测试艺术:策略与实践
  • 一颗B+树可以存储多少数据?
  • 【大数据 复习】第11,12,13,14章
  • SSRF服务端请求伪造
  • STM32三种调试工具CMSIS-DAP、J-Link和ST-Link
  • Renesas MCU使用定时器之实现1ms定时中断
  • redis大key优化
  • 分享一款快速APP功能测试工具
  • [LeetCode] Wiggle Sort
  • gitlab-ci配置详解(一)
  • java第三方包学习之lombok
  • leetcode378. Kth Smallest Element in a Sorted Matrix
  • Linux Process Manage
  • MySQL主从复制读写分离及奇怪的问题
  • Rancher如何对接Ceph-RBD块存储
  • Spring Cloud中负载均衡器概览
  • 番外篇1:在Windows环境下安装JDK
  • 区块链将重新定义世界
  • 让你成为前端,后端或全栈开发程序员的进阶指南,一门学到老的技术
  • 如何优雅地使用 Sublime Text
  • 吐槽Javascript系列二:数组中的splice和slice方法
  • 新手搭建网站的主要流程
  • 一个6年java程序员的工作感悟,写给还在迷茫的你
  • 一些css基础学习笔记
  • 在Docker Swarm上部署Apache Storm:第1部分
  • 【云吞铺子】性能抖动剖析(二)
  • ionic入门之数据绑定显示-1
  • ​Base64转换成图片,android studio build乱码,找不到okio.ByteString接腾讯人脸识别
  • (26)4.7 字符函数和字符串函数
  • (AtCoder Beginner Contest 340) -- F - S = 1 -- 题解
  • (done) NLP “bag-of-words“ 方法 (带有二元分类和多元分类两个例子)词袋模型、BoW
  • (搬运以学习)flask 上下文的实现
  • (二)Pytorch快速搭建神经网络模型实现气温预测回归(代码+详细注解)
  • (二)WCF的Binding模型
  • (二刷)代码随想录第15天|层序遍历 226.翻转二叉树 101.对称二叉树2
  • (附源码)spring boot校园健康监测管理系统 毕业设计 151047
  • (附源码)计算机毕业设计SSM基于健身房管理系统
  • (七)微服务分布式云架构spring cloud - common-service 项目构建过程
  • (一)认识微服务
  • (转)iOS字体
  • .net FrameWork简介,数组,枚举
  • .NET/C# 使用 ConditionalWeakTable 附加字段(CLR 版本的附加属性,也可用用来当作弱引用字典 WeakDictionary)
  • .NET的微型Web框架 Nancy
  • .NET企业级应用架构设计系列之技术选型
  • .Net下C#针对Excel开发控件汇总(ClosedXML,EPPlus,NPOI)
  • .NET值类型变量“活”在哪?
  • @angular/cli项目构建--http(2)