目录
- 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()
- 多个拦截器的执行顺序
-
先按顺序执行所有拦截器的 preHandle方法
-
如果任何一个拦截器preHandle方法返回false。直接跳出不执行目标方法
- 如果当前拦截器prehandler返回为true。则执行下一个拦截器的preHandle方法
- 如果当前拦截器返回为false。直接倒序执行所有已经执行了的拦截器的 afterCompletion方法;
-
所有拦截器都返回True。执行目标方法
-
倒序执行所有拦截器的postHandle方法。
-
前面的步骤有任何异常都会直接倒序触发 afterCompletion
-
页面成功渲染完成以后,也会倒序触发 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来继续响应后续请求
*/
@Override
public boolean preHandle(HttpServletRequest request,
HttpServletResponse response, Object object) throws Exception {
System.out.println("HandlerInterceptor preHandle ....");
return true;
}
/**
* 在控制器方法调用后,解析视图前调用,我们可以对视图和模型做进一步渲染或修改
* 可在modelAndView中加入数据,比如当前时间
*/
@Override
public void postHandle(HttpServletRequest request,HttpServletResponse response,
Object object, ModelAndView modelAndView) throws Exception {
System.out.println("HandlerInterceptor postHandle ....");
}
/**
* 整个请求完成,即视图渲染结束后调用,这个时候可以做些资源清理工作,或日志记录
*/
@Override
public 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 {
@Override
public 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>
- 测试
- 编写controller
@Controller
@RequestMapping("/account")
public class AccountController {
@RequestMapping("/findAccount13")
public String findAccount13(Model model) {
model.addAttribute("msg", "欢迎你 springmvc");
System.out.println("controller的方法执行了......");
return "success";
}
}
- 在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 注意事项
- form表单必须加上
enctype="multipart/form-data"
- method属性取值必须是
Post
- 提供一个文件选择域
<input type="file"/>
3.全局异常处理器
即使对大部分情况都做了预判和异常处理,但程序有时会因为莫名奇妙的原因发生错误,并且错误信息肯定会抛在浏览器页面上,这样用户观感特别不好,所以需要进行全局异常处理。
3.1 异常处理思路
系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理,如下图:
3.2 创建异常处理器
@Component
public class CustomExceptionResolver implements HandlerExceptionResolver {
@Override
public 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>