> 文章列表 > 25.SSM-SpringMVC延续--拦截器与异常处理思想

25.SSM-SpringMVC延续--拦截器与异常处理思想

25.SSM-SpringMVC延续--拦截器与异常处理思想

目录

一、拦截器

(1)拦截器的作用。

(2)拦截器与过滤器的区别。

(3)拦截器的方法。

(4)拦截器的快速入门。

(5)多拦截器。

(5.1)spirng-mvc.xml配置文件。

(5.2)拦截器参考类。

(6)拦截器的拦截范围——重点。

(6.1)拦截器的拦截范围-前端控制器内部。

(6.2)如果静态资源满足拦截条件,但还是请求成功了的原因——并非请求成功。

(7)拦截器的应用-登录权限控制。 

二、SpringMVC异常处理。 

(1)异常处理思路。

(2)异常处理的两种方式。

(2.1) 简单异常处理器。

(2.2)自定义异常处理器。

(3)常见的异常-不重要。


一、拦截器。

(1)拦截器的作用。

(2)拦截器与过滤器的区别。

拦截器(Interceptor)和过滤器(Filter)的作用部分相似,都可以拦截请求并进行处理。但是它们的实现方式不同,主要区别有以下几点:

1.实现方式不同:拦截器是基于 Java 反射机制实现的,而过滤器是基于 Servlet 规范实现的。

2.对象不同:拦截器是针对 Spring MVC 中的请求进行拦截处理的,而过滤器可以对所有的请求进行拦截处理。

3.程序上下文不同:拦截器只能获取到 Spring MVC 的上下文信息,而过滤器可以获取到整个 Servlet 的上下文信息。

4.支持 AOP 的能力:拦截器可以更好地支持 AOP 思想,而过滤器则不能。

5.控制粒度不同:拦截器可以对请求进行细粒度的控制,而过滤器的粒度较粗,只能对请求进行简单的转发或者重定向等操作。

因此,在实际开发中,如果需要对 Spring MVC 中的请求进行一些 AOP 方面的处理或者需要对请求进行更加细粒度的控制,可以使用拦截器;如果只需要对请求进行一些简单的处理(如:字符编码、文件上传、登录验证等),则可以使用过滤器。

(3)拦截器的方法。

(4)拦截器的快速入门。

(5)多拦截器。

(5.1)spirng-mvc.xml配置文件。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context  http://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--1、mvc的注解驱动--><mvc:annotation-driven/><!--2、配置视图解析器--><bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="prefix" value="/"/><property name="suffix" value=".jsp"/></bean><!--3、静态资源权限开放--><mvc:default-servlet-handler/><!--4、组件扫描--><context:component-scan base-package="controller"/><!--5、配置拦截器--><mvc:interceptors><!--拦截器的执行顺序的按照这里的顺序执行的,如果想让哪个先执行,就放前面--><mvc:interceptor><!--对哪些资源执行拦截操作,可配置多个--><mvc:mapping path="/"/><!--配置哪些资源排除拦截操作,可配置多个--><mvc:exclude-mapping path="/user/login"/><bean class="interceptor.MyInterceptor1"/></mvc:interceptor><mvc:interceptor><mvc:mapping path="/"/><bean class="interceptor.MyInterceptor2"/></mvc:interceptor></mvc:interceptors></beans>

(5.2)拦截器参考类。

package interceptor;
public class MyInterceptor1 implements HandlerInterceptor {//运行结果:preHandle......preHandle222222......目标资源执行......postHandle222222......postHandle......afterCompletion222222......afterCompletion......@Override//在目标方法执行之前,执行public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("preHandle......");String param = request.getParameter("param");if ("yes".equals(param)){return true;}else {request.getRequestDispatcher("/error.jsp").forward(request,response);return false;//返回false:不放行;true:放行}}@Override//在目标方法执行之后,视图对象返回之前执行public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {modelAndView.addObject("name","itheima");System.out.println("postHandle......");}@Override//在流程都执行完毕后,执行public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println("afterCompletion......");}
}

(6)拦截器的拦截范围——重点。

<!--SpringMVC的前端控制器--><servlet><servlet-name>DispatcherServlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:spring-mvc.xml</param-value></init-param><load-on-startup>1</load-on-startup><!--表示服务器启动就创建--></servlet><servlet-mapping><servlet-name>DispatcherServlet</servlet-name><!--<url-pattern>/user/*</url-pattern>--><url-pattern>/</url-pattern></servlet-mapping>

笔记:/user/*的映射地址包含/user及其/user/下的。

(6.1)拦截器的拦截范围-前端控制器内部。

进入前端控制器的请求才能被拦截器拦截。(即进入前端控制器后,如果满足拦截器的拦截条件,那么就拦截)。

(6.2)如果静态资源满足拦截条件,但还是请求成功了的原因——并非请求成功。

一般情况下,JSP 中的静态资源(例如图片、CSS、JavaScript 等)请求也是需要经过拦截器的。如果你的 JSP 中引用的静态资源不需要经过拦截器,一般有以下几种情况:

1.静态资源已经被缓存。浏览器在请求静态资源时,如果该资源已经被缓存在本地,那么浏览器就不会发送请求到服务器,因此也就不会经过拦截器。

2.拦截器没有对该静态资源进行拦截。一些拦截器可能只对特定类型的请求进行拦截,例如只对动态请求(如 JSP、Servlet 等)进行拦截,而对静态资源不进行拦截。这种情况需要查看拦截器的配置,了解其拦截规则。

总的来说,无论是动态资源还是静态资源,在请求处理过程中都可能会经过拦截器,这取决于拦截器的配置和运行环境。

笔记:如果静态资源满足拦截条件,但是浏览器正常展示页面,那么就是静态资源被缓存了,浏览器并没有访问服务器,而是使用了缓存的静态资源,你可以试着换一个浏览器访问。

(7)拦截器的应用-登录权限控制。 

package interceptor;
public class PrivilegeInterceptor implements HandlerInterceptor {//这里不需要重写三个,因为只需要用到一个,不用到(不写操作的方法)的就不重写@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {//逻辑:判断用户是否登录,本质:判断session中有没有userHttpSession session = request.getSession();User user = (User) session.getAttribute("user");if (user == null){//没有登录,重定向和转发都可以//request.getRequestDispatcher("/login.jsp").forward(request,response);//可以返回trueresponse.sendRedirect(request.getContextPath()+"/login.jsp");//重定向后不能返回truereturn false;}return true;//放行,访问目标资源}
}

二、SpringMVC异常处理。 

(1)异常处理思路。

原本我们处理异常的方式是try...catch 

(2)异常处理的两种方式。

(2.1) 简单异常处理器。

笔记:程序遇到异常就到springmvc配置文件中找对应的处理方式。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
"><!--1、mvc注解驱动--><mvc:annotation-driven/><!--2、配置视图解析器--><bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="prefix" value="/"/><property name="suffix" value=".jsp"/></bean><!--3、静态资源权限开放--><mvc:default-servlet-handler/><!--4、组件扫描  扫描Controller--><context:component-scan base-package="controller"/><!--5、配置异常处理器--><bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver"><property name="defaultErrorView" value="error"/><property name="exceptionMappings"><map><!--key写异常类的路径,value写出现这个异常后要跳转的页面,没有扩展名是因为配置了视图解析器,有前缀后缀--><entry key="java.lang.ClassCastException" value="error1"/><!--下面这个是自定义异常,也可以不使用前缀后缀(参考前缀后缀-添加规则)--><entry key="exception.MyException" value="redirect:/error2.jsp"/></map></property></bean><!--自定义异常处理器--><!--<bean class="resolver.MyExceptionResolver"/>-->
</beans>

(2.2)自定义异常处理器。

自定义异常处理器类:

public class MyExceptionResolver implements HandlerExceptionResolver {/*参数Exception:异常对象返回值ModelAndView:跳转到错误视图信息*/@Overridepublic ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {ModelAndView modelAndView = new ModelAndView();//左边的对象是否是它右边的类的实例(或子类实例)if (e instanceof MyException){modelAndView.addObject("info","自定义异常");}else if (e instanceof ClassCastException){modelAndView.addObject("info","类转换异常");}modelAndView.setViewName("error");return modelAndView;}
}

配置自定义处理器:

    <!--自定义异常处理器--><bean class="resolver.MyExceptionResolver"/>

(3)常见的异常-不重要。

public class DemoServiceImpl implements DemoService {public void show1() {System.out.println("抛出类型转换异常....");Object str = "zhangsan";Integer num = (Integer)str;}public void show2() {System.out.println("抛出除零异常....");int i = 1/0;}public void show3() throws FileNotFoundException {System.out.println("文件找不到异常....");InputStream in = new FileInputStream("C:/xxx/xxx/xxx.txt");}public void show4() {System.out.println("空指针异常.....");String str = null;str.length();}public void show5() throws MyException {System.out.println("自定义异常....");throw new MyException();//抛出一个异常,可以选择try...catch进行捕获,这里选择把异常抛出去}
}