【 <二> 丹方改良:Spring 时代的 JavaWeb】之 Spring Boot 中的 AOP:实现日志记录与性能监控
<前文回顾>
<今日更新>
一、开篇整活儿
今儿个咱唠唠 Spring Boot 里头的 AOP(面向切面编程)。这玩意儿吧,说大不大,说小不小,整好了是锦上添花,整不好就是火上浇油。你要是刚入门,那可得悠着点儿,别一上来就整得自己“翻车”了。
二、AOP 是啥玩意儿?
AOP 是 Spring 里头的一个高级特性,用来在不修改原有代码的情况下,给程序动态添加功能。比如说,你可以用 AOP 来记录日志、监控性能、处理异常啥的。Spring Boot 里头默认就集成了这玩意儿,用起来贼方便。
1. AOP 的核心概念
AOP 里头有几个核心概念:切面(Aspect)、连接点(Join Point)、通知(Advice)、切点(Pointcut)。
- 切面:就是你要添加的功能,比如说日志记录、性能监控啥的。
- 连接点:就是程序执行过程中的某个点,比如说方法调用、异常抛出啥的。
- 通知:就是切面在连接点执行的动作,比如说在方法调用前记录日志。
- 切点:就是用来匹配连接点的表达式,比如说匹配某个包下的所有方法。
2. AOP 的通知类型
AOP 里头有五种通知类型:
- 前置通知(Before):在连接点之前执行。
- 后置通知(After):在连接点之后执行,不管连接点是否抛出异常。
- 返回通知(AfterReturning):在连接点正常返回后执行。
- 异常通知(AfterThrowing):在连接点抛出异常后执行。
- 环绕通知(Around):在连接点前后都执行,可以控制连接点的执行。
三、用 AOP 实现日志记录
日志记录是 AOP 的经典应用场景。你可以用 AOP 来记录方法的调用信息,方便以后排查问题。
1. 定义切面
首先,你得定义一个切面,用 @Aspect 注解标记。
| Java Code |
| @Aspect @Component public class LoggingAspect { private static final Logger logger = LoggerFactory.getLogger(LoggingAspect.class); } |
这段代码里头,LoggingAspect 是一个切面,@Aspect 注解标记了这个类,@Component 注解让 Spring 管理这个类。
2. 定义切点
然后,你得定义一个切点,用 @Pointcut 注解标记。
| Java Code |
| @Aspect @Component public class LoggingAspect { private static final Logger logger = LoggerFactory.getLogger(LoggingAspect.class); @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} } |
这段代码里头,serviceMethods 是一个切点,execution(* com.example.demo.service.*.*(..)) 表示匹配 com.example.demo.service 包下的所有方法。
3. 定义通知
最后,你得定义通知,用 @Before、@After、@AfterReturning、@AfterThrowing 或 @Around 注解标记。
| Java Code |
| @Aspect @Component public class LoggingAspect { private static final Logger logger = LoggerFactory.getLogger(LoggingAspect.class); @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} @Before("serviceMethods()") public void logMethodCall(JoinPoint joinPoint) { logger.info("调用方法:{}", joinPoint.getSignature().getName()); } } |
这段代码里头,logMethodCall 是一个前置通知,@Before 注解标记了这个方法,JoinPoint 参数用来获取连接点的信息。
四、用 AOP 实现性能监控
性能监控是 AOP 的另一个经典应用场景。你可以用 AOP 来记录方法的执行时间,方便以后优化性能。
1. 定义切面
首先,你得定义一个切面,用 @Aspect 注解标记。
| Java Code |
| @Aspect @Component public class PerformanceAspect { private static final Logger logger = LoggerFactory.getLogger(PerformanceAspect.class); } |
这段代码里头,PerformanceAspect 是一个切面,@Aspect 注解标记了这个类,@Component 注解让 Spring 管理这个类。
2. 定义切点
然后,你得定义一个切点,用 @Pointcut 注解标记。
| Java Code |
| @Aspect @Component public class PerformanceAspect { private static final Logger logger = LoggerFactory.getLogger(PerformanceAspect.class); @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} } |
这段代码里头,serviceMethods 是一个切点,execution(* com.example.demo.service.*.*(..)) 表示匹配 com.example.demo.service 包下的所有方法。
3. 定义通知
最后,你得定义通知,用 @Around 注解标记。
| Java Code |
| @Aspect @Component public class PerformanceAspect { private static final Logger logger = LoggerFactory.getLogger(PerformanceAspect.class); @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} @Around("serviceMethods()") public Object measureMethodExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable { long startTime = System.currentTimeMillis(); Object result = joinPoint.proceed(); long endTime = System.currentTimeMillis(); logger.info("方法 {} 执行时间:{} 毫秒", joinPoint.getSignature().getName(), endTime - startTime); return result; } } |
这段代码里头,measureMethodExecutionTime 是一个环绕通知,@Around 注解标记了这个方法,ProceedingJoinPoint 参数用来控制连接点的执行。
五、AOP 的坑点
1. 切点表达式写错了
AOP 里头,切点表达式写错了,那通知就不起作用了。你要是写错了,那可得好好检查检查。
| Java Code |
| @Pointcut("execution(* com.example.demo.service.*.*(..))") // 写错了 public void serviceMethods() {} |
这段代码里头,execution 写错了,应该是 execution。
当然,写不好表达式,可以问AI啊。
2. 通知顺序不对
AOP 里头,通知顺序不对,那结果就不对了。你要是顺序不对,那可得好好调整调整。
| Java Code |
| @Before("serviceMethods()") public void logMethodCall(JoinPoint joinPoint) { logger.info("调用方法:{}", joinPoint.getSignature().getName()); } @Around("serviceMethods()") public Object measureMethodExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable { long startTime = System.currentTimeMillis(); Object result = joinPoint.proceed(); long endTime = System.currentTimeMillis(); logger.info("方法 {} 执行时间:{} 毫秒", joinPoint.getSignature().getName(), endTime - startTime); return result; } |
这段代码里头,logMethodCall 和 measureMethodExecutionTime 的顺序很重要。
3. 切面没被 Spring 管理
AOP 里头,切面没被 Spring 管理,那通知就不起作用了。你要是没被管理,那可得好好检查检查。
| Java Code |
| @Aspect // 没加 @Component public class LoggingAspect { private static final Logger logger = LoggerFactory.getLogger(LoggingAspect.class); @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} @Before("serviceMethods()") public void logMethodCall(JoinPoint joinPoint) { logger.info("调用方法:{}", joinPoint.getSignature().getName()); } } |
这段代码里头,LoggingAspect 没加 @Component 注解,Spring 不会管理这个类。
六、额外再说一点
对于大多数 Spring Boot 项目,如果你只是简单地使用 AOP 来实现日志记录、事务管理等功能,并且已经引入了 spring-boot-starter-aop 依赖,那么通常不需要显式地使用 @EnableAspectJAutoProxy 注解。Spring Boot 会自动为你处理相关的配置。
然而,如果你有特殊的需求,比如自定义代理创建策略或确保 AOP 支持被启用,那么你可以考虑显式地使用 @EnableAspectJAutoProxy 注解。
专有名词解释
- AOP:面向切面编程,一种编程范式,用来在不修改原有代码的情况下,给程序动态添加功能。
- 切面:AOP 里头的一个概念,表示你要添加的功能。
- 连接点:AOP 里头的一个概念,表示程序执行过程中的某个点。
- 通知:AOP 里头的一个概念,表示切面在连接点执行的动作。
- 切点:AOP 里头的一个概念,表示用来匹配连接点的表达式。
- 前置通知:AOP 里头的一种通知类型,在连接点之前执行。
- 后置通知:AOP 里头的一种通知类型,在连接点之后执行。
- 返回通知:AOP 里头的一种通知类型,在连接点正常返回后执行。
- 异常通知:AOP 里头的一种通知类型,在连接点抛出异常后执行。
- 环绕通知:AOP 里头的一种通知类型,在连接点前后都执行。
- JoinPoint:AOP 里头的一个接口,用来获取连接点的信息。
- ProceedingJoinPoint:AOP 里头的一个接口,用来控制连接点的执行。
相关文章:
【 <二> 丹方改良:Spring 时代的 JavaWeb】之 Spring Boot 中的 AOP:实现日志记录与性能监控
<前文回顾> 点击此处查看 合集 https://blog.csdn.net/foyodesigner/category_12907601.html?fromshareblogcolumn&sharetypeblogcolumn&sharerId12907601&sharereferPC&sharesourceFoyoDesigner&sharefromfrom_link <今日更新> 一、开篇整…...
多模态大模型训练范式演进与前瞻
本文从多模态大模型相关概念出发,并以Flamingo 模型为例,探讨了基于多模态大模型训练的演进与前瞻。新一代训练范式包括统一架构、数据工程革新和动态适应机制,以提升跨模态推理能力和长视频理解。 多模态大模型 定义 什么是多模态大模型&…...
游戏引擎学习第187天
看起来观众解决了上次的bug 昨天遇到了一个相对困难的bug,可以说它相当棘手。刚开始的时候,没有立刻想到什么合适的解决办法,所以今天得从头开始,逐步验证之前的假设,收集足够的信息,逐一排查可能的原因&a…...
HarmonyOS NEXT 关于鸿蒙的一多开发(一次开发,多端部署) 1+8+N
官方定义 定义:一套代码工程,一次开发上架,多端按需部署。 目标:支撑开发者快速高效的开发支持多种终端设备形态的应用,实现对不同设备兼容的同时,提供跨设备的流转、迁移和协同的分布式体验。 什么是18…...
SAP-ABAP:OData 协议深度解析:架构、实践与最佳应用
OData 协议深度解析:架构、实践与最佳应用 一、协议基础与核心特性 协议定义与目标 定位:基于REST的开放数据协议,标准化数据访问接口,由OASIS组织维护,最新版本为OData v4.01。设计哲学:通过统一资源标识符(URI)和HTTP方法抽象数据操作,降低异构系统集成复杂度。核心…...
当Kafka化身抽水马桶:论组件并发提升与系统可用性的量子纠缠关系
《当Kafka化身抽水马桶:论组件并发提升与系统可用性的量子纠缠关系》 引言:一场OOM引发的血案 某个月黑风高的夜晚,监控系统突然发出刺耳的警报——我们的数据发现流水线集体扑街。事后复盘发现:Kafka集群、Gateway、Discovery服…...
Dify+ollama+vanna 实现text2sql 智能数据分析 -01
新鲜出炉-今天安装vanna踩过的坑 今天的任务是安装vanna这个工具,因为dify中自己写的查询向量数据库和执行sql这两步太慢了大概要20S,所以想用下这个工具,看是否会快一点。后面会把这个vanna封装成一个工具让dify调用。 环境说明 我是在本…...
【Python实用技巧】OS模块详解:文件与目录操作的瑞士军刀
大家好,我是唐叔!今天咱们来聊聊Python中那个被低估的"老黄牛"——os模块。这个模块看似简单,但却是每个Python开发者都绕不开的利器。就像我常说的:“不会用os模块的Python程序员,就像不会用筷子的美食家”…...
动态内存分配与内存对齐
在C语言及其他低级编程语言中,内存管理是一个至关重要的主题。动态内存分配和内存对齐是确保程序高效和稳定运行的关键因素。本文将深入探讨动态内存分配的原理,内存对齐的概念,并解释它们如何共同影响程序的性能和资源利用。 一、动态内存分配简介 1.1 动态内存分配的概念…...
C 标准库 – 头文件
1️⃣ <fenv.h> 简介 <fenv.h> 提供了用于控制和检查浮点运算行为的宏和函数。它为浮点环境提供了精细的控制,允许设置舍入模式、捕获浮点异常等。通过 <fenv.h>,程序员可以: 控制浮点舍入模式,指定不同的舍入…...
Redis的基础,经典,高级问题解答篇
目录 一,基础 二,经典 缓存雪崩: 1. Redis事务的原子性 2. 与MySQL事务的区别 1. 主从复制原理 2. 哨兵模式故障转移流程 3. 客户端感知故障转移 三,高级 一,基础 Redis的5种基础数据类型及使用场景…...
uniapp uni-swipe-action滑动内容排版改造
在uniapp开发中 默认的uni-swipe-action滑动组件 按钮里的文字都是横排的 不能换行的 如果是在一些小屏设备 比如PDA这种,同时按钮文字又都是4个字 多按钮的情况 就会发现滑动一下都直接满屏了 观看体验都不好 但默认的官方组件又没有样式的设置,下面就告…...
电脑卡怎么办?提升电脑流畅的方法
电脑已经成为我们工作、学习和娱乐不可或缺的伙伴。然而,随着使用时间的增长,许多用户会遇到电脑运行变慢、卡顿的情况,这不仅影响了工作效率,也大大降低了使用体验。本文将为大家分析电脑卡顿的常见原因,并提供一套实…...
SpringBoot报错解决方案
org.apache.tomcat.util.http.fileupload.impl.SizeLimitExceededException: the request was rejected because its size (31297934) exceeds the configured maximum (10485760) 文件上传大小超过限制...
知识表示方法之二:产生式表示法(Production System)
有关知识表示方法之一:一阶谓词逻辑的内容可以看我的文章:https://blog.csdn.net/lzm12278828/article/details/146541478 一、定义 “产生式”这一术语是有美国数学家博斯特(E.Post)在1943年首先提出来的,他根据串替代规则提出了一种称为波斯特机的计算模型,模型中的每…...
SQL Server中账号权限
目录标题 查看现有权限授予权限拒绝权限撤销权限角色管理 在SQL Server中管理账号权限主要通过以下几个关键步骤: 查看现有权限 可以使用系统视图来查看账号的权限,比如 sys.database_permissions 视图,示例查询如下: SELECT d…...
软件需求未明确非功能性指标(如并发量)的后果
软件需求未明确非功能性指标(如并发量)可能带来的严重后果包括:系统性能下降、用户体验恶化、稳定性降低、安全风险增加、后期维护成本高企。其中,系统性能下降尤为显著。当软件系统在设计和开发阶段未明确并发量需求时࿰…...
VScode-i18n-ally-Vue
参考这篇文章,做Vue项目的国际化配置,本篇文章主要解释,下载了i18n之后,该如何对Vscode进行配置 https://juejin.cn/post/7271964525998309428 i18n Ally全局配置项 Vscode中安装i18n Ally插件,并设置其配置项&#…...
Spring Boot项目快速创建-开发流程(笔记)
主要流程: 前端发送网络请求->controller->调用service->操纵mapper->操作数据库->对entity数据对象赋值->返回前端 前期准备: maven、mysql下载好 跟学视频,感谢老师: https://www.bilibili.com/video/BV1gm4…...
车架号查询车牌号接口如何用Java对接
一、什么是车架号查询车牌号接口? 车架号查询车牌号接口,即传入车架号,返回车牌号、车型编码、初次登记日期信息。车架号又称车辆VIN码,车辆识别码。 二、如何用Java对接该接口? 下面我们以阿里云接口为例࿰…...
npm : 无法加载文件 C:\Program Files\nodejs\npm.ps1,因为在此系统上禁止运行脚本的处理方法
1、安装了node.js后,windows powershell中直接输入npm,然后就报错 2、出现原因:权限不够 系统禁用了脚本的执行,所以我们在windows powershell输入npm -v的时候,就会报上面的错误。 3、解决 Set-ExecutionPolicy Un…...
【java笔记】泛型、包装类、自动装箱拆箱与缓存机制
一、泛型:类型安全的基石 1. 泛型的本质与原理 Java 泛型(Generics)是 JDK 5 引入的特性,通过类型参数化实现代码的通用性。泛型类、接口和方法允许在定义时声明类型参数(如 T、E、K、V),这些…...
数仓开发那些事(11)
某神州优秀员工:一闪,领导说要给我涨米。 一闪:。。。。(着急的团团转) 老运维:Oi,两个吊毛,看看你们的hadoop集群,健康度30分,怎么还在抽思谋克?…...
自动化测试【Python3.7+Selenium3】
1、自动化测试环境搭建之selenium3安装 方法1:cmd环境下,用pip install selenium (速度很慢,不推荐) 方法2:下载selenium安装包手动安装 下载地址:https://pypi.org/project/selenium/ 在解压…...
从零开始完成冒泡排序(0基础)——C语言版
文章目录 前言一、冒泡排序的基本思想二、冒泡排序的执行过程(一)第一轮排序(二)第二轮排序(三)第三轮排序(四)第四轮排序 三、冒泡排序的代码实现(C语言)&am…...
工业级POE交换机:助力智能化与自动化发展
随着工业互联网、物联网(IoT)和自动化技术的快速发展,网络设备在工业领域的应用日益广泛。然而,在严苛环境下,传统网络设备往往难以应对复杂的温湿度变化、电磁干扰和供电不稳定等挑战。为同时满足数据传输与供电一体化…...
使用ZYNQ芯片和LVGL框架实现用户高刷新UI设计系列教程(第五讲)
在上一讲我们讲解了按键回调函数的自定义函数的用法,这一讲继续讲解回调函数的另一种用法。 首先我们将上一讲做好的按键名称以及自定义回调事件中的按键名称修改,改为默认模式为“open”当点击按键时进入回调函数将按键名称改为“close”,具…...
Burp Suite Professional 2024版本安装激活指南
文章目录 burpsuite简介Burp Suite的主要组件:Burp Suite的版本使用场景 下载地址使用教程 burpsuite简介 Burp Suite 是一个广泛使用的网络安全测试工具,特别是在Web应用程序安全领域。它主要用于发现和修复Web应用中的安全漏洞,特别适用于渗…...
【c++深入系列】:类与对象详解(上)
🔥 本文专栏:c 🌸作者主页:努力努力再努力wz 💪 今日博客励志语录: 你仰望的星辰并非遥不可及,而是跋涉者脚印的倒影;你向往的远方未必需要翅膀,只要脚下始终有路&#x…...
6、进程理论和简单进程创建
一、了解进程推荐看这个视频,很详细 1、概念 进程(Process)程序的运行过程,是系统进行资源分配和调度的独立单元程序的运行过程:多个不同程序 并发,同一个程序同时执行多个任务。 就需要很多资源来实现这个过程。 每个进程都有一…...
