Spring的过滤器、拦截器、切面区别及案例分析

Spring的过滤器、拦截器、切面 三者的区别,以及对应案例分析

在这里插入图片描述

一、三者的实现方式

1.1 过滤器 xxxFilter

过滤器的配置比较简单,直接实现Filter接口即可,也可以通过@WebFilter注解实现对特定URL的拦截,Filter接口中定义了三个方法:

  • init(): 该方法在容器启动初始化过滤器时被调用,它在过滤器的整个生命周期只会被调用一次
  • doFilter(): 容器中的每一次请求都会调用该方法,其中的FilterChain用来调用下一个过滤器
  • destory(): 当容器销毁过滤器实例时调用该方法,在过滤器的整个生命周期也只会被调用一次

1.2 拦截器 xxxInterceptor

拦截器它是链式调用,一个应用中可以同时存在多个拦截器,一个请求也可以触发多个拦截器,而每个拦截器的调用会依据它的声明顺序依次执行。实现HandlerInterceptor接口,该接口定义了三个方法:

  • preHandle(): 这个方法将在请求处理之前调用,如果该方法返回false,将视为当前请求结束;
  • postHandle(): 在Controller中的方法调用之后,DispatcherServlet返回渲染视图之前被调用;
  • afterComplete(): 在整个请求结束之后,DispatcherServlet渲染了对应视图之后执行。

1.3 切面 xxxAspect

切面注解能够在任何不想改原来代码的地方添加处理逻辑,可以根据自定义注解定义切面。

  • @Aspect:定义在切面实现类上的注解
  • @Pointcut:定义切入点
  • @Before:切面方法执行之前的增强
  • @After:切面方法执行之后的增强,不管抛异常还是正常退出都执行的增强
  • @AfterReturning:切面方法执行之后的增强,方法正常退出时执行
  • @AfterThrowing:切面方法执行之后的增强,抛出异常后执行
  • @Around:环绕增强,包围一个连接点的增强,可获取执行前信息,可修改执行后数据

二、三者的区别

过滤器:
无法获取请求要访问的类、方法以及参数;
可以获取原始的Http请求与响应。

拦截器:
可以获取请求访问的类、方法;
但是无法获取请求参数的值。

切面:
可以获取访问的类、方法以及参数值;
无法获取Http原始的请求与响应的对象。

=================== 请求处理顺序 ===================
过滤器 -> 拦截器 -> 切面
=================== 报错处理顺序 ===================
切面 -> @ControllerAdvice -> 拦截器 -> 过滤器

三、三者的实现案例

3.1 过滤器:MyFilter

package com.wyw.learn.namebindingvsaop;import org.springframework.stereotype.Component;import javax.servlet.*;
import java.io.IOException;/*** @author name:    silk* @version 1.0* @description: TODO* @date 2024/4/22 20:18*/
@Component
public class MyFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {Filter.super.init(filterConfig);}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {String filterVal = "这是我设置的过滤器参数值";servletRequest.setAttribute("myFilterParam", filterVal);filterChain.doFilter(servletRequest, servletResponse);}@Overridepublic void destroy() {Filter.super.destroy();}
}

3.2 拦截器:MyInterceptor

package com.wyw.learn.namebindingvsaop;import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;/*** @author name:    silk* @version 1.0* @description: TODO* @date 2024/4/22 20:22*/
@Component
public class MyInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {String interceptorVal = "这是我设置的拦截器参数值";request.setAttribute("myInterceptorParam", interceptorVal);return HandlerInterceptor.super.preHandle(request, response, handler);}@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {HandlerInterceptor.super.postHandle(request, response, handler, modelAndView);}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {HandlerInterceptor.super.afterCompletion(request, response, handler, ex);}
}

另外,拦截器需要在配置类中规定拦截哪些路径

package com.wyw.learn.namebindingvsaop;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;/*** @author name:    silk* @version 1.0* @description: TODO* @date 2024/4/22 21:49*/
@Configuration   // 配置项
public class MyConfig implements WebMvcConfigurer {@Autowiredprivate MyInterceptor myInterceptor;@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(myInterceptor).addPathPatterns("/**");   // 拦截所有URL;}
}

3.3 切面:MyAspect

切面注解:

package com.wyw.learn.namebindingvsaop;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface AopAnno {
}

切面逻辑:

package com.wyw.learn.namebindingvsaop;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;
import java.util.Arrays;/*** @author name:    silk* @version 1.0* @description: TODO* @date 2024/4/22 15:39*/
@Component
@Aspect
public class MyAspect {@Pointcut(value = "@annotation(com.wyw.learn.namebindingvsaop.AopAnno)")public void pointCut() {}@Before("pointCut()")public void doProcess(JoinPoint joinPoint) {Object obj = joinPoint.getArgs().length == 0 ? null : joinPoint.getArgs()[0];if (obj == null) {return;}// 获取request对象,注意这已不是原始的请求ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest request = attributes.getRequest();String aspectVal = "这是我设置的切面参数值";request.setAttribute("myAspectParam", aspectVal);// 通过切面给参数属性赋值Arrays.stream(obj.getClass().getDeclaredFields()).forEach(field -> {field.setAccessible(true);if ("description".equals(field.getName())) {try {field.set(obj, "这是我给请求参数属性设置的切面参数值");} catch (IllegalAccessException e) {e.printStackTrace();}}});}
}

3.4 测试类:MyTestController

package com.wyw.learn.namebindingvsaop;import com.itheima.reggie.entity.Dish;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;/*** @author name:    silk* @version 1.0* @description: TODO* @date 2024/4/22 20:35*/
@RestController
@RequestMapping("/myTest")
public class MyTestController {@Autowiredprivate HttpServletRequest request;@PostMapping("/test")@AopAnnopublic void testDemo(@RequestBody Dish dish) {Object myFilter = request.getAttribute("myFilterParam");Object myInterceptor = request.getAttribute("myInterceptorParam");Object myAspect = request.getAttribute("myAspectParam");String description = dish.getDescription();System.out.println("请求中过滤器参数值是:" + myFilter);System.out.println("请求中拦截器参数值是:" + myInterceptor);System.out.println("请求中切面参数值是:" + myAspect);System.out.println("请求参数的Description属性的参数值是:" + description);}
}

3.5 测试结果

在这里插入图片描述

附:

附一篇之前写的博客,分析过滤器和拦截器的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2978690.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

工作记录:vue-grid-layout 修改 margin 导致 item 高度剧烈变化

问题 用 vue-gird-layout 时发现,当改变 margin 值时,item 的尺寸也会跟着变化。 如下图:row height 和每个 item 的 h 都保持不变。修改 margin-y,item 的实际高度也跟着变了: 原因 研究了一番,发现原…

如何查看全球历史影像

目录 示例 2024年3月28日 2022年9月21日 2021年11月3日 2020年11月18日 2019年5月15日 2017年2月27日 2016年12月20日 如何在ArcGIS中加载 如何查看全球历史影像,今天给大家分享一个可以在线直接访问查看全球历史影像的网站(网址见文末&#xf…

flutter 设置启屏页 flutter_native_splash 坑记录

flutter_native_splash | Flutter packageCustomize Flutters default white native splash screen with background color and splash image. Supports dark mode, full screen, and more.https://pub.dev/packages/flutter_native_splash 发现一直白屏 原因是 代码中 下面…

前端路由的实现原理

当谈到前端路由时,指的是在前端应用中管理页面导航和URL的机制。前端路由使得单页应用(Single-Page Application,SPA)能够在用户与应用交互时动态地加载不同的视图,而无需每次都重新加载整个页面。 在前端开发中&…

vue3+elementui-plus实现无限递归菜单

效果图 实现方式是&#xff1a;通过给定的数据结构层数来动态生成多级菜单 menu.vue<template><el-menu:default-active"activeIndex"class"el-menu-demo"mode"horizontal"select"handleSelect"background-color"#f8f…

Echarts的饼图有哪些配置项,一文告诉你。

Echarts中的饼图&#xff08;Pie Chart&#xff09;有以下一些常用的配置项&#xff1a; 配置项一 Echarts中的饼图&#xff08;Pie Chart&#xff09;有以下一些常用的配置项&#xff1a; title&#xff1a;饼图的标题配置&#xff0c;包括text&#xff08;标题文本&#xf…

冈萨雷斯数字图像处理资源(课后习题答案+代码+图片)

冈萨雷斯数字图像处理相关资源整理&#xff0c;资源全部来源互联网&#xff0c;方便大家下载 冈萨雷斯数字图像处理相关资源整理 课后习题 冈萨雷斯数字图像处理源代码

程序猿成长之路之数据挖掘篇——朴素贝叶斯

朴素贝叶斯是数据挖掘分类的基础&#xff0c;本篇文章将介绍一下朴素贝叶斯算法 情景再现 以挑选西瓜为例&#xff0c;西瓜的色泽、瓜蒂、敲响声音、触感、脐部等特征都会影响到西瓜的好坏。那么我们怎么样可以挑选出一个好的西瓜呢&#xff1f; 分析过程 既然挑选西瓜有多个…

静态缺省路由的应用

静态缺省路由的应用 缺省路由是目的地址全零的特殊路由&#xff0c;可以由路由协议自动生成&#xff0c;也可以由手动配置。手动配置缺省路由&#xff0c;可以简化网络的配置&#xff0c;称为静态缺省路由。如果报文的目的地址无法匹配路由表中的任何一项&#xff0c;路由器将…

查看项目go代码cpu利用率

1.代码添加&#xff1a; "net/http"_ "net/http/pprof"第二步&#xff0c;在代码开始运行的地方加上go func() {log.Println(http.ListenAndServe(":6060", nil))}() 2.服务器上防火墙把6060打开 3.电脑安装&#xff1a;Download | Graphviz …

合并二叉树【c++】

#include <iostream> #include <vector> using namespace std;//双链表节点结构 typedef struct treeNode {int value;struct treeNode* left;struct treeNode* right;treeNode(int x) : value(x), left(nullptr), right(nullptr) {} } TreeNode;TreeNode* mergeTr…

BFS解决FloodFill算法:(Leetcode:733. 图像渲染)

题目链接&#xff1a;733. 图像渲染 - 力扣&#xff08;LeetCode&#xff09; 使用广度优先遍历算法解决该问题&#xff1a; 从初始位置开始搜索&#xff0c;初始位置符合条件就入栈&#xff0c;并修改初始位置值。初始位置出栈。 再从初始位置开始广度优先搜索&#xff08;…

免 Administrator 权限安装软件

以欧路词典为例, 从官网下载的安装包 https://www.eudic.net/v4/en/app/download 直接运行会弹出 UAC 提示需要管理员权限. 一个词典而已, 为啥要管理员权限呢? 答案是安装程序默认使用的安装路径是 C:\Program Files\ 这就不难理解了. 对于这种不需要其他额外权限的软件, 可以…

【大模型开源篇1】彦宏您怎么看LLaMA3的开源

Meta LLaMA是Meta公司开源的大模型&#xff0c;作为大模型开源界得鼻祖&#xff0c; 刚刚发布LLaMA3。从ChatGPT 拉开了大模型竞赛的序幕&#xff0c;Meta 选择了开源&#xff0c;至此大模型也开始百花齐放的时期&#xff0c;但是开源模型一直无法超过必源模型&#xff0c;如今…

图像超分辨率技术在AI去衣中的应用探索

在数字图像处理领域&#xff0c;图像超分辨率&#xff08;Super-Resolution, SR&#xff09;技术一直是研究的热点之一。该技术旨在从低分辨率的图像中恢复出高分辨率的图像&#xff0c;以提供更清晰、更丰富的细节信息。近年来&#xff0c;随着人工智能&#xff08;AI&#xf…

【电控笔记5.8】数字滤波器设计流程频域特性

数字滤波器设计流程&频域特性 2HZ : w=2pi2=12.56 wc=2*pi*5; Ts=0.001; tf_lpf =

windows实现文件共享

新建一块磁盘作为共享磁盘 开启该磁盘文件共享 访问该共享磁盘 上传镜像到该共享磁盘 在目标主机上查看已上传的共享镜像

【nmap】基于python实现nmap端口扫描

nmap下载&#xff1a;nmap 将所在路径放入环境变量&#xff1a; D:\Program Files\Nmap\zenmap\bin 安装模块 pip install python-nmap python代码实现&#xff1a; import nmapdef scan(host, min_port, max_port):port_range str(min_port) - str(max_port)nm nmap.Po…

Python | Leetcode Python题解之第43题字符串相乘

题目&#xff1a; 题解&#xff1a; class Solution:def multiply(self, num1: str, num2: str) -> str:if num1 "0" or num2 "0":return "0"m, n len(num1), len(num2)ansArr [0] * (m n)for i in range(m - 1, -1, -1):x int(num1[i…

【基础算法】双指针

1.移动零 移动零 思路&#xff1a; 利用双指针算法 cur&#xff1a;从左往右扫描数组&#xff0c;遍历数组 dest&#xff1a;处理好的区间包括dest dest初始化为-1&#xff0c;因为刚开始dest前应该没有非零元素。 即将非零元素移到dest之前即可 class Solution { public…