spring AOP切面及日志记录实现

article/2025/9/21 13:56:34

目录

1.什么是AOP切面

2.理解AOP

3.AOP实例

1.自定义注解

2.创建一个切面类

3.将自定义注解标注在测试接口上


1.什么是AOP切面

AOP(Aspect Oriented Programming),面向切面思想,是Spring的三大核心思想之一。

在项目中经常会有些系统性的需求,例如权限校验,日志记录,统计等,这时我们就可以通过AOP切面去实现。

有多少业务代码就需要写多少重复校验和日志记录,这显然是不合理的,我们可以把这些重复操作抽离出来,写成公共的方法。

这样,代码冗余解决了,但是每个地方都要手动去调用还是很麻烦,有没有更好的方式呢?这就要用到我们的AOP(面向切面编程),AOP将权限校验,日志记录等非业务代码提取出来,和业务代码完全分离,并寻找节点切入业务代码中

2.理解AOP

简单理解AOP,AOP主要做三件事

  1. 在哪里切入,也就是日志记录等非业务代码在哪些业务代码中执行。
  2. 在什么时候切入,是在业务代码执行前还是后。
  3. 切入后做什么事情,比如权限校验,日志记录等

 理解如下图

一些概念理解:

Pointcut:切点,决定处理如权限校验、日志记录等在何处切入业务代码中(即织入切面)。切点分为execution方式和annotation方式。前者可以用路径表达式指定哪些类织入切面,后者可以指定被哪些注解修饰的代码织入切面。
Advice:处理,包括处理时机和处理内容。处理内容就是要做什么事,比如校验权限和记录日志。处理时机就是在什么时机执行处理内容,分为前置处理(即业务代码执行前)、后置处理(业务代码执行后)等。
Aspect:切面,即Pointcut和Advice。
Joint point:连接点,是程序执行的一个点。例如,一个方法的执行或者一个异常的处理。在 Spring AOP 中,一个连接点总是代表一个方法执行。
Weaving:织入,就是通过动态代理,在目标对象方法中执行处理内容的过程。

3.AOP实例

  1. 自定义一个注解OperationLogAnnotation
  2. 创建一个切面类,切点设置为拦截标注OperationLogAnnotation的方法,截取传参,进行日志记录
  3. 将OperationLogAnnotation标注在测试接口上

具体的实现步骤如下:

1.自定义注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface OperationLogAnnotion {//操作类型@NonNullOperationType type();//操作对象@NonNullOperationObjectType objType();}
public enum OperationType {CREATE("create"),UPDATE("update"),DELETE("delete"),ADD("add");private String type;OperationType(String type){this.type=type;}public String getType() {return type;}public void setType(String type) {this.type = type;}
}
public enum OperationObjectType {USER("user"),//项目PROJECT("project"),//页面管理PROJECT_PAGE("project page"),//人员管理ROLE_USER("role user"),//角色页面管理ROLE_PAGE("role page");private String objType;OperationObjectType(String objType){this.objType=objType;}public String getObjType() {return objType;}public void setObjType(String objType) {this.objType = objType;}
}

2.创建一个切面类

@Aspect
@Component
public class LogAdvice implements ApplicationListener<ContextRefreshedEvent> {private BlockingQueue<OperationLog> operationLogQ = new LinkedBlockingDeque<>();@Autowiredprivate IOperationLogService operationLogService;private static final Logger LOGGER = LogManager.getLogger();@Pointcut("@annotation(com.pwd.springdemo.aop.annotation.OperationLogAnnotion)")public void logAdvicePointcut(){}@AfterReturning(pointcut = "logAdvicePointcut()  && @annotation(operationLogAnnotion)", returning = "returnResult")public void logAdvice(JoinPoint joinPoint, com.pwd.springdemo.aop.annotation.OperationLogAnnotion operationLogAnnotion, Object returnResult){try {if (!(returnResult instanceof ResultInfo)) {return;}//filter failed requestResultInfo result = (ResultInfo) returnResult;if (result.getCode() != ResultCode.SUCCESS.getCode()) {return;}HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();Object[] args = joinPoint.getArgs();String requestParams = JSONObject.toJSONString(args);String userIp = request.getHeader("UserIp");User user = (User) request.getSession().getAttribute("user");String userId = null;if (user != null) {userId = user.getUserId();}String type = operationLogAnnotion.type().getType();String objType = operationLogAnnotion.objType().getObjType();Signature signature = joinPoint.getSignature();String functionName=signature.getName();OperationLog log = new OperationLog(userId, userIp, functionName,type + " " + objType, requestParams);operationLogQ.add(log);} catch (Exception e) {LOGGER.error("operation log aspect error", e);}}/*** consume queue 2 save db*/private void saveLogs() {while (true) {try {OperationLog log = operationLogQ.poll(5, TimeUnit.MILLISECONDS);if (log != null) {operationLogService.addLog(log);}} catch (Exception e) {LOGGER.error("insert operation log to db error", e);}}}@Overridepublic void onApplicationEvent(ContextRefreshedEvent event) {ExecutorService pool = new ThreadPoolExecutor(1, 1,0L, TimeUnit.SECONDS,new LinkedBlockingQueue<>(), new ThreadFactoryBuilder().setNameFormat("Save-Logs-%d").setDaemon(true).build());pool.submit(new Runnable() {@Overridepublic void run() {saveLogs();}});}
}

3.将自定义注解标注在测试接口上

    @ResponseBody@OperationLogAnnotion(type = OperationType.DELETE,objType = OperationObjectType.USER)@RequestMapping(value = "deleteUserInfoById", method = RequestMethod.POST)public ResultInfo deleteUserInfoById(@RequestBody JSONObject jsonObject) {Integer id = jsonObject.getInteger("id");LOGGER.info("id:" + id);userService.deleteUserInfoById(id);return ResultInfo.success(null);}


http://chatgpt.dhexx.cn/article/CEI1djUN.shtml

相关文章

Aop切面自定义注解的使用

一&#xff1a;功能简介 本文主要记录如何使用aop切面的方式来实现日志记录功能。 主要记录的信息有: 操作人&#xff0c;方法名&#xff0c;参数&#xff0c;运行时间&#xff0c;操作类型(增删改查)&#xff0c;详细描述&#xff0c;返回值。 二&#xff1a;项目结构图 三…

AOP切面注解

一.前言 在以前的项目中&#xff0c;很少去关注spring aop的具体实现与理论&#xff0c;只是简单了解了一下什么是aop具体怎么用&#xff0c;看到了一篇博文写得还不错&#xff0c;就转载来学习一下&#xff0c;博文地址&#xff1a;http://www.cnblogs.com/xrq730/p/4919025.h…

AOP切面执行顺序

文章目录 一. 概述二. 讲述1. 单切面中各通知方法的执行顺序2. 多切面中各通知方法的执行顺序3. 多切面的通知方法中抛出异常 参考资料 一. 概述 本文主要讲述以下几点 单AOP切面时&#xff0c;各通知方法的执行顺序。多AOP切面时&#xff0c;多切面的执行顺序和各通知方法的执…

spring aop切面执行顺序

spring aop切面执行顺序 切面执行顺序 现有切面1、切面2对同一个切点按先后顺序执行&#xff08;切面1先于切面2执行&#xff09; 切面1&#xff1a;Before、After、AfterReturnning、AfterThrowing、Around 执行前、Around 正常执行、Around 异常执行、Around 执行后切面2&am…

一文带你搞定AOP切面

摘要&#xff1a;AOP在spring中又叫“面向切面编程”&#xff0c;是对传统我们面向对象编程的一个补充&#xff0c;主要操作对象就是“ 切面”&#xff0c; 可以简单的理解它是贯穿于方法之中&#xff0c;在方法执行前、执行时、执行后、返回值后、异常后要执行的操作。 本文分…

SpringBoot AOP切面实现

文章目录 一、AOP简介二、AOP体系与概念三、AOP实例1、创建SpringBoot工程2、添加依赖3、AOP相关注解3.1、Aspect3.2、Pointcut3.2.1、execution()3.2.2、annotation() 3.3、Around3.4、Before3.5、After3.6、AfterReturning3.7、AfterThrowing 一、AOP简介 AOP&#xff08;As…

AOP切面编程的理解

一、什么是Spring的AOP&#xff1f; AOP在spring中又叫“面向切面编程”&#xff0c;它可以说是对传统我们面向对象编程的一个补充&#xff0c;从字面上顾名思义就可以知道&#xff0c;它的主要操作对象就是“切面”&#xff0c;所以我们就可以简单的理解它是贯穿于方法之中&a…

AOP切面使用

一、主要设计注解&#xff1a; Aspect After before Pointcut Around pom文件引入 <!--用于aop切面编程--> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> </dependency> 二、AOP核心…

AOP面向切面

1.什么是Spring的AOP? AOP又叫"面向切面编程",是对传统的面向对象编程的一个补充,主要的操作对象就是"切面 ",可以简单的理解它是贯穿于方法之中,在方法执行前、执行时、执行后、返回值后、异常后要执行的操作。 相当于将我们原本一条线执行的程序在中间切…

【JavaEE】Spring AOP (面向切面)详解

目录&#xff1a; 1. 什么是 Spring AOP&#xff1f;1.1 AOP1.2 使用 AOP 的场景 2. AOP 组成2.1 切面&#xff08;Aspect&#xff09;2.2 连接点&#xff08;Join Point&#xff09;2.3 切点&#xff08;Pointcut&#xff09;2.4 通知&#xff08;Advice&#xff09; 3. AOP 概…

斜杠,反斜杠说明

/ 斜杠 \反斜杠 在window中都用斜杠 反斜杠是用来转译字符串的 eg: \"a\" 输出"a"

斜杠、反斜杠、双斜杠、反双斜杠的区别和使用方法及范围

背景 这边我就找了两篇大神写的文章&#xff0c;讲得非常清晰明了。文章主要讲了一些历史缘故和我们面对各种斜杠时的疑惑。 斜杠’/’ 和反斜杠’’ 深入探讨正斜杠和反斜杠 概念 1. 斜杠"/"是URL地址中用到的分隔符&#xff0c;并且在linux系统中的文件路径也是…

glob.glob()之返回路径的正反斜杆问题

Windows环境下用一个反斜杠就行 绝对路径&#xff1a;D:\PyCharm_code\pytorch_study\xxx 相对路径&#xff1a;.\cifar10_train\**\*.png 以下是踩过的坑&#xff1a;记录下

正反斜杠的区别_正斜杠( / )和反斜杠( \ )的区别

反斜杠“\”是电脑出现了之后为了表示程序设计里的特殊含义才发明的专用标点。所以除了程序设计领域外&#xff0c;任何地方都不应该使用反斜杠。 如何区分正反斜杠 英语&#xff1a;"/" 英文是forward slash, “\" 是backward slash 形象些比喻的话&#xff0c…

正斜杠 “/” 与反斜杠 “\”辨析

文章目录 1. 正斜杠 / / /2. 反斜杠 \ \backslash \3. 正斜杠与反斜杠的区别4. 注意 注意&#xff0c; / / / 为正斜杠(forward slash)&#xff0c;而 \ \backslash \ 为反斜杠(backward slash)。 1. 正斜杠 / / / 斜线是斜线标点符号 / / /。曾经用于标记句点和逗号的斜…

斜杠'/' 和反斜杠'\'

斜杠’/‘和反斜杠’\’ 2019-1-21 引言&#xff1a;从大一进入信息专业&#xff0c;正式接触计算机、代码也有几年了。一开始迷迷糊糊学Ascii码&#xff0c;很多特殊字符都需要转义&#xff0c;比如换行符\n&#xff0c;自那时起我就拎不清转义符是斜杠还是反斜杠&#xff0c;…

全面了解 Python 中的反斜杆

本文全面介绍了 Python 中反斜杆(\)的用法&#xff0c;包括原始字符串和普通字符串&#xff0c;repr() 和 str() ,\ 作为转义符&#xff0c;\ 作为续行符&#xff0c;\ 在字符串转义和正则表达式转义中的过程及注意事项等。阅读本文预计 6 min. 全面了解 Python 中的反斜杆 1. …

教你认识正斜杠(/)与反斜杠(\)

正斜杠&#xff0c;又称左斜杠&#xff0c;符号是 “/” ; 反斜杠&#xff0c;也称右斜杠&#xff0c;符号是 “” 。 经常很迷惑正斜杠与反斜杠到底有何区别&#xff1f;以下是一些总结: 背景了解 &#xff1a; DOS路径&#xff1a; C:\WINDOWS\SETTING …这是反斜杠的作用后…

微信支付,二维码图片解析

微信支付&#xff1a; 后台返回的是数据流&#xff1b; 开始这样&#xff0c;但是不行&#xff0c; 解决&#xff1a;在请求里面加入 ‘responseType’: ‘blob’ , 转换&#xff1a;附上base64转图片 //base64转换base64ImgtoFile(dataurl, filename file) {let arr data…

Apache里如何将图片解析成PHP

首先&#xff0c;如果没有安装PHP&#xff0c;先安装PHP yum install -y php然后进入网站根目录&#xff0c;如果不记得网站根目录&#xff0c;可以去配置文件里找 我的是/mnt/z 所以进入这个目录下&#xff0c;新建一个i.jpg文件 在浏览器里查看这个文件&#xff0c;存在错误…