【JavaEE进阶】——Spring AOP
目录
🚩Spring AOP概述
🚩Spring AOP快速⼊⻔
🎓引入AOP依赖
🎓编写AOP程序
🚩Spring AOP 详解
🎓Spring AOP核⼼概念
🎓通知类型
🎓@PointCut
🎓切⾯优先级 @Order
🎓切点表达式
📝execution表达式
📝@annotation
🎈⾃定义注解 @MyAspect
🎈切⾯类
🎈添加⾃定义注解
🚩Spring AOP的实现⽅式(常⻅⾯试题)
🚩Spring AOP概述
Spring框架的第一大核心是IOC(控制反转),接下来我们学习第二大核心——AOP.
什么是AOP?
Aspect Oriented Programming(⾯向切⾯编程)和面向对象编程不是互斥的关系,也不是面向对象编程的升级,是面向对象编程的补充。什么是⾯向切⾯编程呢? 切⾯就是指某⼀类特定问题, 所以AOP也可以理解为⾯向特定⽅法编程. 什么是⾯向特定⽅法编程呢? ⽐如上个章节学习的"登录校验", 就是⼀类特定问题. 登录校验拦截器, 就 是对"登录校验"这类问题的统⼀处理. 所以, 拦截器也是AOP的⼀种应⽤.AOP是⼀种思想, 拦截器是AOP 思想的⼀种实现. Spring框架实现了这种思想, 提供了拦截器技术的相关接⼝. 同样的, 统⼀数据返回格式和统⼀异常处理, 也是AOP思想的⼀种实现.简单来说: AOP是⼀种思想, 是对某⼀类事情的集中处理(用户登录校验,结果统一返回,异常统一处理)
拦截器作⽤的维度是URL(⼀次请求和响应), @ControllerAdvice 应⽤场景主要是全局异常处理 (配合⾃定义异常效果更佳), 数据绑定, 数据预处理.
AOP作⽤的维度更加细致(可以根据包、类、⽅法 名、参数等进⾏拦截), 能够实现更加复杂的业务逻辑.
一个项目需要开发很多的业务功能,有一些业务的执行效率是很低的,需要对接口进行优化。第一步就需要定位出执行耗时比较长的业务方法,然后针对该业务方法进行优化。如何定位呢?我们就需要统计当前项目中每一个业务方法的执行耗时,我们需要在每个业务方法运行前和运行后,记录下方法的开始时间和结束时间,两者之差就是这个方法的耗时。但是一个项目中有很多的方法和类,我们需要记录每个方法的执行时长,需要在每个方法中记录,这种执行效率是比较低的,耗时比较长,
AOP就可以做到在不改动这些原始⽅法的基础上, 针对特定的⽅法进⾏功能的增强.AOP的作⽤:在程序运⾏期间在不修改源代码的基础上对已有⽅法进⾏增强(⽆侵⼊性: 解耦)
🚩Spring AOP快速⼊⻔
需求: 统计图书系统各个接⼝⽅法的执⾏时间
🎓引入AOP依赖
在pom.xml⽂件中添加配置
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>
🎓编写AOP程序
记录Controller中每个⽅法的执⾏时间
@Slf4j
@Component
@Aspect
public class TimeRecordAspect {/*** 记录时间*/@Around("execution(* com.example.cl.controller.*.*(..))")//对哪些方法生效//@Around表示侵入程序的时机,表示环绕,目标函数执行前后都会执行public Object TimeRecord(ProceedingJoinPoint joinPoint) throws Throwable {//记录开始时间long start=System.currentTimeMillis();//目标函数执行前//执行目标方法Object proceed=joinPoint.proceed();//执行目标函数//记录结束时间long end=System.currentTimeMillis();//目标函数执行后//日志打印耗时时间log.info("耗时时间:"+(end-start)+" ms");return proceed;}//AOP这种写法作用的维度是 方法//拦截器 作用的维度是 url 接口
}
在进行用户登录,我们看到该方法耗时时间:234ms.
对程序进⾏简单的讲解:
- @Aspect: 标识这是⼀个切⾯类
- @Around: 环绕通知, 在⽬标⽅法的前后都会被执⾏. 后⾯的表达式表⽰对哪些⽅法进⾏增强.
- ProceedingJoinPoint.proceed() 让原始⽅法执⾏
我们通过AOP⼊⻔程序完成了业务接⼝执⾏耗时的统计.通过上⾯的程序, 我们也可以感受到 AOP⾯向切⾯编程的⼀些优势 :
- • 代码⽆侵⼊: 不修改原始的业务⽅法, 就可以对原始的业务⽅法进⾏了功能的增强或者是功能的改变
- • 减少了重复代码
- • 提⾼开发效率
- • 维护⽅便
🚩Spring AOP 详解
🎓Spring AOP核⼼概念
🎈切点(Pointcut)切点(Pointcut), 也称之为"切⼊点"Pointcut 的作⽤就是提供⼀组规则 (使⽤ AspectJ pointcut expression language 来描述), 告诉程序对 哪些⽅法来进⾏功能增强.@Around("execution(* com.example.cl.controller.*.*(..))") 这就是切点表达式🎈连接点(Join Point)满⾜切点表达式规则的⽅法, 就是连接点. 也就是可以被AOP控制的⽅法以⼊⻔程序举例, 所有 com.example.demo.controller 路径下的⽅法, 都是连接点UserController/BookController 中的⽅法都是连接点📝切点表达式和连接点的关系:连接点是满足切点表达式的元素,切点可以看作是保存了众多连接点的一个集合连接点——路径下里面的所有的方法切点表达式——路径🎈通知(Advice)通知就是具体要做的⼯作, 指哪些重复的逻辑,也就是共性功能(最终体现为⼀个⽅法)⽐如上述程序中记录业务⽅法的耗时时间, 就是通知
在AOP⾯向切⾯编程当中, 我们把这部分重复的代码逻辑抽取出来单独定义, 这部分代码就是通知的内容.🎈切⾯(Aspect)切⾯(Aspect) = 切点(Pointcut) + 通知(Advice)通过切⾯就能够描述当前AOP程序需要针对于哪些⽅法, 在什么时候执⾏什么样的操作切⾯既包含了通知逻辑的定义, 也包括了连接点的定义
切⾯所在的类, 我们⼀般称为切⾯类(被@Aspect注解标识的类)
🎓通知类型
Spring中AOP的通知类型有以下⼏种:
- @Around: 环绕通知, 此注解标注的通知⽅法在⽬标⽅法前, 后都被执⾏
- @Before: 前置通知, 此注解标注的通知⽅法在⽬标⽅法前被执⾏
- @After: 后置通知, 此注解标注的通知⽅法在⽬标⽅法后被执⾏, ⽆论是否有异常都会执⾏
- @AfterReturning: 返回后通知, 此注解标注的通知⽅法在⽬标⽅法后被执⾏, 有异常不会执⾏
- @AfterThrowing: 异常后通知, 此注解标注的通知⽅法发⽣异常后执⾏
@Aspect
@Slf4j
public class AspectDemo {/*** 前置通知*/@Before("execution(* com.cl.springaop.controller.*.*(..))")public void doBefore(){log.info("AspectDemo do before......");}//是否发生异常,都会执行(再执行方法前)优先级小于around/*** 后置通知*/@After("execution(* com.cl.springaop.controller.*.*(..))")public void doAfter(){log.info("AspectDemo do after......");}//是否发生异常,都会执行(再执行方法后) 优先级小于around/***添加环绕通知*///around必须有返回值,需要将目标返回值返回,否则会没有返回值@Around("execution(* com.cl.springaop.controller.*.*(..))")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {log.info("AspectDemo do around before......");Object result=null;result=joinPoint.proceed();log.info("AspectDemo do around after.......");return result;}//发生异常就不会返回结果/*** 返回后通知*/@AfterReturning("execution(* com.cl.springaop.controller.*.*(..))")public void doAfterReturning(){log.info("AspectDemo do AfterReturning......");}//没有异常执行/*** 抛出异常后通知*/@AfterThrowing("execution(* com.cl.springaop.controller.*.*(..))")public void doAfterThrowing(){log.info("AspectDemo do AfterThrowing......");}//有异常执行
}
写一些测试程序:
@RestController
@Slf4j
@RequestMapping("/test")
public class TestController {@RequestMapping("/t1")public String t1(){log.info("执行t1方法");return "t1";}@RequestMapping("/t2")public String t2(){log.info("执行t2方法.......");int a=10/0;return "t2";}
}
1. 正常运⾏的情况
程序正常运⾏的情况下, @AfterThrowing 标识的通知⽅法不会执⾏从上图也可以看出来,@Around 标识的通知⽅法包含两部分, ⼀个"前置逻辑", ⼀个"后置逻辑". 其中"前置逻辑" 会先于 @Before 标识的通知⽅法执⾏, "后置逻辑" 会晚于 @After 标识的通知⽅法执⾏![]()
2.异常时的情况
程序发⽣异常的情况下:
- • @AfterReturning 标识的通知⽅法不会执⾏, @AfterThrowing 标识的通知⽅法执⾏了
- • @Around 环绕通知中原始⽅法调⽤时有异常,通知中的环绕后的代码逻辑也不会在执⾏了(因为原始⽅法调⽤出异常了)
注意事项:
- • @Around 环绕通知需要调⽤ ProceedingJoinPoint.proceed() 来让原始⽅法执⾏, 其他 通知不需要考虑⽬标⽅法执⾏.
- • @Around 环绕通知⽅法的返回值, 必须指定为Object, 来接收原始⽅法的返回值, 否则原始⽅法执⾏完毕, 是获取不到返回值的.
- • ⼀个切⾯类可以有多个切点.
🎓@PointCut
@Pointcut("execution(* com.cl.springaop.controller.*.*(..))")public void pt(){};
@Pointcut("execution(* com.cl.springaop.controller.*.*(..))")public void pt(){};/*** 前置通知*/@Before("pt()")public void doBefore(){log.info("AspectDemo do before......");}//是否发生异常,都会执行(再执行方法前)优先级小于around/*** 后置通知*/@After("pt()")public void doAfter(){log.info("AspectDemo do after......");}//是否发生异常,都会执行(再执行方法后) 优先级小于around
当切点定义使⽤private修饰时, 仅能在当前切⾯类中使⽤,当其他切⾯类也要使⽤当前切点定义时, 就需要把 private改为public.引⽤⽅式为: 全限定类名.⽅法名()![]()
@Component @Aspect @Slf4j public class AspectDemo2 {@Before("com.cl.springaop.aspect.AspectDemo.pt()")public void doBefore(){log.info(" AspectDemo2 do before......");} }
🎓切⾯优先级 @Order
为防⽌⼲扰, 我们把AspectDemo这个切⾯先去掉(把 @Component 注解去掉就可以 )


存在多个切⾯类时, 默认按照切⾯类的类名字⺟排序:
- • @Before 通知:字⺟排名靠前的先执⾏
- • @After 通知:字⺟排名靠前的后执⾏
但这种⽅式不⽅便管理, 我们的类名更多还是具备⼀定含义的.Spring 给我们提供了⼀个新的注解, 来控制这些切⾯通知的执⾏顺序: @Order
按照4,3,2的顺序在先执行。
@Order 注解标识的切⾯类, 执⾏顺序如下:
- • @Before 通知:数字越⼩先执⾏
- • @After 通知:数字越⼤先执⾏

🎓切点表达式
上⾯的代码中, 我们⼀直在使⽤切点表达式来描述切点. 下⾯我们来介绍⼀下切点表达式的语法.切点表达式常⻅有两种表达⽅式
- 1. execution(RR):根据⽅法的签名来匹配
- 2. @annotation(RR) :根据注解匹配
📝execution表达式
execution(< 访问修饰符 > < 返回类型 > < 包名 . 类名 . ⽅法 ( ⽅法参数 )> < 异常 >)

📝@annotation
实现步骤:
- 1. 编写⾃定义注解
- 2. 使⽤ @annotation 表达式来描述切点
- 3. 在连接点的⽅法上添加⾃定义注解
UserController
@Component
@RestController
@Slf4j
@RequestMapping("/user")
public class UserController {@RequestMapping("/u1")public String u1(){log.info("执行u1方法");return "u1";}//由于没有引入注解,所以没有@RequestMapping("/u2")public String u2(){log.info("执行u2方法");return "u2";}
}
TestController
@RestController
@Slf4j
@RequestMapping("/test")
public class TestController {@RequestMapping("/t1")public String t1(){log.info("执行t1方法");return "t1";}@RequestMapping("/t2")public String t2(){log.info("执行t2方法.......");int a=10/0;return "t2";}
}
🎈⾃定义注解 @MyAspect
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface MyAspect {
}
🎈切⾯类
@Aspect
@Component
@Slf4j
public class MyAspectDemo {@Around("@annotation(com.cl.springaop.config.MyAspect)")public Object doAround(ProceedingJoinPoint joinPoint) {log.info("MyAspectDemo do around before.....");Object result= null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("发生异常,e:"+e);}log.info("MyAspectDemo do around after.....");return result;}
}
@Around("@annotation(org.springframework.web.bind.annotation.RequestMapping)")public Object doAround2(ProceedingJoinPoint joinPoint) {log.info("RequestMapping do around before.....");Object result= null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("发生异常,e:"+e);}log.info("RequestMapping do around after.....");return result;}
🎈添加⾃定义注解


对testController中的t2测试
没有打印结果。
🚩Spring AOP的实现⽅式(常⻅⾯试题)
- 1. 基于注解 @Aspect
- 2. 基于Spring API (通过xml配置的⽅式, ⾃从SpringBoot ⼴泛使⽤之后, 这种⽅法⼏乎看不到了)
- 3. 基于代理来实现(更加久远的⼀种实现⽅式, 写法笨重, 不建议使⽤)
背挺直一点,向前靠一点。
相关文章:

【JavaEE进阶】——Spring AOP
目录 🚩Spring AOP概述 🚩Spring AOP快速⼊⻔ 🎓引入AOP依赖 🎓编写AOP程序 🚩Spring AOP 详解 🎓Spring AOP核⼼概念 🎓通知类型 🎓PointCut 🎓切⾯优先级 Ord…...
Python - conda使用大全
如何使用Conda? 环境 创建环境 conda create -n spider_env python3.10.11查看环境 conda env listconda info -e激活环境 conda activate spider_env退出环境 conda deactivate删除环境 conda env remove -n spider_env包 导出包 说明:导出当前虚拟…...
ASPICE在汽车软件开发中的作用
ASPICE是一个专门为汽车软件开发过程而设计的评估和改进框架。它基于ISO/IEC 15504标准,为汽车供应商提供了一个评估和改进其软件开发流程的方法。ASPICE的目标是确保软件开发过程的一致性和可预测性,从而提高软件的质量和可靠性。 ASPICE的实施对汽车软…...

亚马逊云科技 re:Inforce 2024中国站大会
亚马逊云科技 re:Inforce 2024中国站大会 - 生成式AI时代的全面安全,将于7月25日本周四在北京富力万丽酒店揭幕...
Lottie:动态动画的魔法棒
文章目录 引言官网链接Lottie 的原理基础使用1. 导出动画2. 引入 Lottie 库3. 加载和播放动画 高级使用1. 动画控制2. 交互性3. 自定义动画例子:交互式按钮动画 优缺点优点缺点 结语 引言 Lottie 是 Airbnb 开源的一个动画库,它允许设计师在 Adobe Afte…...
IPython使用技巧整理
IPython 是一个增强的 Python 交互式 shell,它提供了许多便利的功能,比如自动补全、魔术命令、对象内省等。以下是 IPython 的一些使用技巧和示例,结合您提供的列表数据,我将给出一些相关的使用示例。 1. 自动补全(Tab…...

C#数组复习
一、一维数组 using System.Collections; using System.Collections.Generic; using UnityEngine;public class ShuZu : MonoBehaviour {#region 知识点一 基本概念//数组是存储一组相同类型数据的集合//数组分为 一维、二维、交错数组//一般情况 一维数组 就简称为数组#en…...
无人机之在农业上的用途
随着无人机技术的发展,农业现代化也迎来了崭新局面,田间随处可见无人机矫健的身影。当农业遇上科技,变革正悄然进行。农业无人机主要应用于农业、种植业、林业等行业。在使用过程中,其功能和作用并不单一,一般用于种植…...
opengaussdb在oepnEuler上安装
安装前提: 软件环境:openEuler 20.03LTS 个人开发者最低配置2核4G,推荐配置4核8G 数据库版本:openGauss-5.0.2-openEuler-64bit-all.tar.gz 数据库下载地址: https://docs-opengauss.osinfra.cn/zh/docs/5.0.0/docs/In…...

一些和颜色相关网站
1.中国传统色 2.网页颜色选择器 3.渐变色网站 4.多风格色卡生成 5.波浪生成 6.半透明磨砂框 7.色卡组合...
Linux系统编程-文件系统
目录 什么是Linux文件系统 文件系统的职责 存储介质抽象 inode:文件系统的核心 文件分配策略 目录结构 文件系统布局 日志和恢复机制 目录权限 粘滞位(t位): 硬链接和符号链接 硬链接的特点: 创建硬链接: 符号链接的…...

【解决】ubuntu20.04 root用户无法SSH登陆问题
Ubuntu root用户无法登录的问题通常可以通过修改SSH配置文件和系统登录配置来解决。 修改SSH配置文件 sudo vim /etc/ssh/sshd_config 找到 PermitRootLogin 设置,并将其值更改为 yes 以允许root用户通过SSH登录 保存并关闭文件之后,需要重启SSH服务…...

(前缀和) LeetCode 238. 除自身以外数组的乘积
一. 题目描述 原题链接 给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&…...

【JVM基础05】——组成-能不能解释一下方法区?
目录 1- 引言:方法区概述1-1 方法区是什么?(What)1-2 为什么用方法区?方法区的作用 (Why) 2- ⭐核心:详解方法区(How)2-1 能不能解释一下方法区?2-2 元空间内存溢出问题2-3 什么是常量池?2-4 运行时常量池 …...

前端:Vue学习-3
前端:Vue学习-3 1. 自定义指令2. 插槽2.1 插槽 - 后备内容(默认值)2.2 插槽 - 具名插槽2.3 插槽 - 作用域插槽 3. Vue - 路由3.1 路由模块封装3.2 声明式导航 router-link 高亮3.3 自定义匹配的类名3.4 声明式导肮 - 跳转传参3.5 Vue路由 - 重…...

npm 安装报错(已解决)+ 运行 “wue-cli-service”不是内部或外部命令,也不是可运行的程序(已解决)
首先先说一下我这个项目是3年前的一个项目了,中间也是经过了多个人的修改惨咋了布置多少个人的思想,这这道我手里直接npm都安装不上,在网上也查询了多种方法,终于是找到问题所在了 问题1: 先是npm i 报错在下面图片&…...

江苏科技大学24计算机考研数据速览,有专硕复试线大幅下降67分!
江苏科技大学(Jiangsu University of Science and Technology),坐落在江苏省镇江市,是江苏省重点建设高校,江苏省人民政府与中国船舶集团有限公司共建高校,国家国防科技工业局与江苏省人民政府共建高校 &am…...

20分钟上手新版Skywalking 9.x APM监控系统
Skywalking https://skywalking.apache.org/ Skywalking是专为微服务、云原生和基于容器的(Kubernetes)架构设计的分布式系统性能监控工具。 Skywalking关键特性 ● 分布式跟踪 ○ 端到端分布式跟踪。服务拓扑分析、以服务为中心的可观察性和API仪表板。…...

【07】LLaMA-Factory微调大模型——微调模型导出与微调参数分析
上文介绍了如何对微调后的模型进行使用与简单评估。本文将介绍对微调后的模型进行导出的过程。 一、llama-3微调后的模型导出 首先进入虚拟环境,打开LLaMA-Factory的webui页面 conda activate GLM cd LLaMA-Factory llamafactory-cli webui 之后,选择…...

动态路由协议 —— EIGRP 与 OSPF 的区别
EIGRP(增强内部网关路由协议)和 OSPF(开放式最短路径优先)是两种最常见的动态路由协议,主要是用来指定路由器或交换机之间如何通信。将其应用于不同的情况下,可提高速率、延迟等方面的性能。那么它们之间到…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
<6>-MySQL表的增删查改
目录 一,create(创建表) 二,retrieve(查询表) 1,select列 2,where条件 三,update(更新表) 四,delete(删除表…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...

Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序
一、开发环境准备 工具安装: 下载安装DevEco Studio 4.0(支持HarmonyOS 5)配置HarmonyOS SDK 5.0确保Node.js版本≥14 项目初始化: ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...

Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...
鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/
使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题:docker pull 失败 网络不同,需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词
Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵,其中每行,每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid,其中有多少个 3 3 的 “幻方” 子矩阵&am…...

优选算法第十二讲:队列 + 宽搜 优先级队列
优选算法第十二讲:队列 宽搜 && 优先级队列 1.N叉树的层序遍历2.二叉树的锯齿型层序遍历3.二叉树最大宽度4.在每个树行中找最大值5.优先级队列 -- 最后一块石头的重量6.数据流中的第K大元素7.前K个高频单词8.数据流的中位数 1.N叉树的层序遍历 2.二叉树的锯…...

AI病理诊断七剑下天山,医疗未来触手可及
一、病理诊断困局:刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断",医生需通过显微镜观察组织切片,在细胞迷宫中捕捉癌变信号。某省病理质控报告显示,基层医院误诊率达12%-15%,专家会诊…...