【JavaEE】Spring AOP详解

一.AOP的定义.

  • Aspect Oriented Programming(面向切面编程)概括的来说AOP是一种思想, 是对某一类事情的集中处理
    • 什么是面向切面编程呢? 切面就是指某一类特定问题, 所以AOP也可以理解为面向特定方法编程.
    • 什么是面向特定方法编程呢? 比如上个博客文章所写的"登录校验", 就是一类特定问题. 登录校验拦截器, 就是对"登录校验"这类问题的统一处理. 所以, 拦截器也是AOP的一种应用. AOP是一种思想, 拦截器是AOP思想的一种实现.Spring框架实现了这种思想, 提供了拦截器技术的相关接口.同样的, 统一数据返回格式和统一异常处理, 也是AOP思想的一种实现.
    • AOP是一种思想, 它的实现方法有很多, 有Spring AOP,也有AspectJ、CGLIB等. Spring AOP是其中的一种实现方式.
  • AOP的作用:在程序运行期间在不修改源代码的基础上对已有方法进行增强(无侵入性: 解耦)
  • 举个使用AOP的例子:
    • 现在我们手上有一个项目, 项目中开发了很多的业务功能, 但是现在项目的运行效率较低(项目运行的时间过长) 现在需要对项目的时长进行评估, 找出哪一个模块消耗的时间长, 传统的方法就是对每个模块的每个方法进行计时,算出这个模块需要的总时间. 这种方法是可以解决问题的, 但一个项目中会包含很多业务模块, 每个业务模块又有很多接口, ⼀个接口又包含很多方法, 如果我们要在每个业务方法中都记录方法的耗时, 对于程序员而言, 会增加很多的工作量. AOP就可以做到在不改动这些原始方法的基础上, 针对特定的方法进行功能的增强

二.AOP的快速入门.

  • AOP的pom.xml依赖:
		<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId></dependency>

编写一个AOP的实现,用来计算Controller中每个方法的运行时间

@Component
@Aspect
@Slf4j
public class AspectDemo1 {
@Around("execution(* com.tuanzi.aop.controller.*.*(..))")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {//规范情况下,要有返回值,否则可能会造成controller中没有返回结果log.info("执行了around before方法");Object result = null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("执行 doAround的目标函数, 内部出现异常");throw new RuntimeException(e);}log.info("执行了around after方法");return result;}
}

运行结果:
在这里插入图片描述

先对程序进行简单解释:

  1. @Aspect: 用来标识这是一个切面类.
  2. @Around: 环绕通知, 在目标方法执行前后都会执行, 后面的表达式表示对哪些方法进行增强.
  3. joinPoint.proceed(): 表示让目标方法(Controller中的方法)执行

整个代码可以分为三部分:
在这里插入图片描述

  • 我们通过AOP入门程序完成了业务接口执行耗时的统计.通过上面的程序, 我们也可以感受到AOP面向切面编程的一些优势:
    • 代码无侵入: 不修改原始的业务方法, 就可以对原始的业务方法进行了功能的增强或者是功能的改变
    • 减少了重复代码
    • 提高开发效率
    • 维护方便

三.AOP详解.

3.1.1切点(Pointcut)

  • Pointcut 的作用就是提供一组规则 (使用 AspectJ pointcut expression language 来描述), 告诉程序对
    哪些方法来进行功能增强.

execution(* com.example.demo.controller..(…)) 就是切点表达式.

3.1.2连接点(Join Point)

  • 满足切点表达式规则的方法就是连接点. 也就是可以被AOP控制的方法.
    比如TestController中的test1和test2方发.

3.1.3通知(advice)

  • 通知就是具体要做的工作, 指哪些重复的逻辑, 也就是共性功能(最终体现为一个方法). 比如上述程序中记录业务方法的消耗时间,就是通知.

3.1.4切面(Aspect)

  • 切面(Aspect) = 切点(Pointcut) + 通知(Advice)
  • 通过切面就能够描述当前AOP程序需要针对于哪些方法, 在什么时候执行什么样的操作. 切面既包含了通知逻辑的定义, 也包括了连接点的定义.
  • 切面所在的类, 我们一般称为切面类(被@Aspect注解标识)

举个例子来彻底理解上面的这几个概念.
现在有一则通知, 22级软件工程的全体同学 , 在上午的10点在教学楼前开会
针对上面的一句话:

  • 切点就是: 22级的全体同学.
  • 连接点就是: 张三,李四, 王五…
  • 通知就是: 在上午10点在教学楼前开会.
  • 切面就是: 22级的全体同学,在上午10点在教学楼前开会.

3.2 通知类型

  • Spring的通知类型有一下几种:
    • @Around: 环绕通知, 此注解标注的通知方法在目标方法前, 后都被执行.
    • @Before: 前置通知, 此注解标注的通知方法在目标方法前被执行.
    • @After: 后置通知,此注解标注的通知方法在目标方法后被执行, 无论是否异常都会执行.
    • @AfterReturning: 返回后通知,此注解标注的通知方法在目标方法后被执行, 有异常不会执行.
    • @AfterThrowing: 异常后通知, 此注解标注的通知方法发生异常后执行.
@Component
@Aspect
@Slf4j
public class AspectDemo1 {@Pointcut("execution(* com.tuanzi.aop.controller.*.*(..))")public void pointCut() {}@Before("pointCut()")public void doBefore(JoinPoint joinPoint) {log.info("执行了before方法");}@After("execution(* com.tuanzi.aop.controller.*.*(..))")public void doAfter(JoinPoint joinPoint) {log.info("执行了after方法");}@Around("execution(* com.tuanzi.aop.controller.*.*(..))")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {//规范情况下,要有返回值,否则可能会造成controller中没有返回结果log.info("执行了around before方法");Object result = null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("执行 doAround的目标函数, 内部出现异常");throw new RuntimeException(e);}log.info("执行了around after方法");return result;}@AfterReturning("execution(* com.tuanzi.aop.controller.*.*(..))")public void doAfterReturning(JoinPoint joinPoint) {log.info("执行了afterReturning方法");}@AfterThrowing("execution(* com.tuanzi.aop.controller.*.*(..))")public void doAfterThrowing(JoinPoint joinPoint) {log.info("执行了afterThroeing方法");}}
  • 正常情况:
    在这里插入图片描述
  • 异常情况:
    在这里插入图片描述

3.3.@PointCut

上面代码存在一个问题, 就是存在大量重复的切点表达式 execution(* com.example.demo.controller..(…)) ,Spring提供了 @PointCut 注解, 把公共的切点表达式提取出来, 需要用到时引用该切入点表达式即可.
在这里插入图片描述

  • 修改后的代码:
    @Pointcut("execution(* com.tuanzi.aop.controller.*.*(..))")public void pointCut() {}@Before("pointCut()")public void doBefore(JoinPoint joinPoint) {log.info("执行了before方法");}

注意事项:
当切点定义使用private修饰时, 仅能在当前切面类中使用, 当其他切面类也要使用当前切点定义时, 就需
要把private改为public. 引用方式为: 全限定类名.方法名()

@Slf4j
@Aspect
@Component
public class AspectDemo2 {//前置通知@Before("com.example.demo.aspect.AspectDemo.pt()")public void doBefore() {log.info("执⾏ AspectDemo2 -> Before ⽅法");}
}

3.4 切面的优先级(@Order)

  • 当一个项目中有多个切面类, 并且这些切面类的多个切点都匹配到同一个目标方法中,当目标方法运行的时候,所有切面类的通知方法都会执行,那这些切面执行的先后顺序该如何定义?
  • 不加任何措施的执行结果:
    在这里插入图片描述
    通过上述程序的运行结果, 可以看出:
    • 存在多个切面类时, 默认按照切面类的类名字母排序:
    • @Before 通知:字母排名靠前的先执行
    • @After 通知:字母排名靠前的后执行

但这种方式不方便管理, 我们的类名更多还是具备一定含义的. Spring 给我们提供了一个新的注解, 来控制这些切⾯通知的执行顺序: @Order

  • @Order的使用方法:
@Aspect
@Component
@Order(2)
public class AspectDemo2 {//...代码省略
}
@Aspect
@Component
@Order(1)
public class AspectDemo3 {//...代码省略
}
@Aspect
@Component
@Order(3)
public class AspectDemo4 {//...代码省略
}
  • 加上@Order的执行结果:
    在这里插入图片描述

  • 通过上述程序的运行结果, 得出@Order 注解标识的切面类, 执行顺序如下:

    • @Before 通知:数字越小先执行
    • @After 通知:数字越大先执行
    • @Order 控制切面的优先级, 先执行优先级较高的切面, 再执行优先级较低的切面, 最终执行目标方法

在这里插入图片描述

3.5切点表达式

切点表达式常见有两种表达方式:

  1. execution(RR):根据方法的签名来匹配
  2. @annotation(RR) :根据注解匹配

3.5.1 execution表达式

execution() 是最常用的切点表达式, 用来匹配方法, 语法为:

execution(<访问修饰符> <返回类型> <包名.类名.方法(方法参数)> <异常>)

切点表达式⽀持通配符表达:

  1. * :匹配任意字符,只匹配一个元素(返回类型, 包, 类名, 方法或者方法参数)
    a. 包名使用 * 表示任意包(一层包使用⼀个*)
    b. 类名使⽤ * 表示任意类
    c. 返回值使⽤ * 表示任意返回值类型
    d. ⽅法名使⽤ * 表示任意方法
    e. 参数使⽤ * 表示一个任意类型的参数
  2. :匹配多个连续的任意符号, 可以通配任意层级的包, 或任意类型, 任意个数的参数
    a. 使用 … 配置包名,标识此包以及此包下的所有子包
    b. 可以使用 … 配置参数,任意个任意类型的参数

切点表达式示例:

  • TestController 下的 public修饰, 返回类型为String 方法名为t1, 无参方法

    execution(public String com.example.demo.controller.TestController.t1())

  • 省略访问修饰符

    execution(String com.example.demo.controller.TestController.t1())

  • 匹配所有返回类型

    execution(* com.example.demo.controller.TestController.t1())

  • 匹配TestController 下的所有无参方法

    execution(* com.example.demo.controller.TestController.*())

  • 匹配TestController 下的所有方法

    execution(* com.example.demo.controller.TestController.*(…))

  • 匹配controller包下所有的类的所有方法

    execution(* com.example.demo.controller..(…))

  • 匹配所有包下面的TestController

    execution(* com…TestController.*(…))

  • 匹配com.example.demo包下, 子孙包下的所有类的所有方法

    execution(* com.example.demo…*(…))

3.5.2@annotation

  • execution表达式更适用有规则的, 如果我们要匹配多个无规则的方法呢, 比如:TestController中的t1()
    和UserController中的u1()这两个方法.

  • 这个时候我们使用execution这种切点表达式来描述就不是很方便了. 我们可以借助自定义注解的方式以及另一种切点表达式 @annotation 来描述这一类的切点.

  • 实现步骤:

    1. 编写自定义注解
    2. 使用 @annotation 表达式来描述切点
    3. 在连接点的方法上添加自定义注解
  • 自定义注解@MyAspect:
    创建一个注解类(和创建Class文件一样的流程, 选择Annotation就可以了)
    在这里插入图片描述

@Target(ElementType.METHOD)
//说明这个注解只能作用在方法上
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAspect {
}

代码简单说明:

  1. @Target 标识了 Annotation 所修饰的对象范围, 即该注解可以用在什么地方.
    常用取值:
    • ElementType.TYPE: 用于描述类、接口(包括注解类型) 或enum声明
    • ElementType.METHOD: 描述方法
    • ElementType.PARAMETER: 描述参数
    • ElementType.TYPE_USE: 可以标注任意类型
  2. @Retention 指Annotation被保留的时间长短, 标明注解的生命周期. @Retention 的取值有三种:
      1. RetentionPolicy.SOURCE:表示注解仅存在于源代码中, 编译成字节码后会被丢弃. 这意味着在运行时无法获取到该注解的信息, 只能在编译时使用. 比如 @SuppressWarnings , 以及lombok提供的注解 @Data , @Slf4j
      1. RetentionPolicy.CLASS:编译时注解. 表示注解存在于源代码和字节码中, 但在运行时会被丢弃. 这意味着在编译时和字节码中可以通过反射获取到该注解的信息, 但在实际运行时无法获取. 通常用于一些框架和工具的注解.
      1. RetentionPolicy.RUNTIME:运行时注解. 表示注解存在于源代码, 字节码和运行时中. 这意味着在编译时, 字节码中和实际运行时都可以通过反射获取到该注解的信息. 通常用于一些需要在运行时处理的注解, 如Spring的 @Controller @ResponseBody.
  • 切面类如何书写:
@Aspect
@Component
@Slf4j
public class MyAspectDemo {/*** 切面之间互不影响,* 如果一个方法既实现了RequestMapping接口,又使用了MyAspect注解,则两个切面的功能都会实现*///作用的范围是标识了这个注解的全部方法@Around("@annotation(com.tuanzi.aop.config.MyAspect)")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {log.info("MyAspectDemo doAround before");Object proceed = joinPoint.proceed();log.info("MyAspectDemo doAround after");return proceed;}//对所有实现requestMapping接口的方法生效@Around("@annotation(org.springframework.web.bind.annotation.RequestMapping)")public Object doAround2(ProceedingJoinPoint joinPoint) throws Throwable {log.info("RequestMapping doAround before");Object proceed = joinPoint.proceed();log.info("RequestMapping  doAround after");return proceed;}
}
  • 如何使用自定义注解
@MyAspect
@RequestMapping("/t1")
public String t1() {return "t1";
}
@MyAspect
@RequestMapping("/u1")
public String u1(){return "u1";
}

四.总结.

Spring AOP的实现方式

  1. 基于注解 @Aspect (参考上述课件内容)
  2. 基于自定义注解 (参考自定义注解 @annotation 部分的内容)
  3. 基于Spring API (通过xml配置的方式, 自从SpringBoot 广泛使用之后, 这种方法几乎看不到了)
    想了解的可以参考链接 https://cloud.tencent.com/developer/article/2032268
  4. 基于代理来实现(更加久远的一种实现方式, 写法笨重, 不建议使用)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

百度搜索框制作HTML+CSS

样品图 自制效果图&#xff08;附注释&#xff09; <!DOCTYPE html> <html lang"en"><head><!-- 定义文档的字符编码为UTF-8&#xff0c;以支持中文等多语言字符 --><meta charset"UTF-8" /><!-- 设置页面在不同设备上的…

掌握Perl命令行:深入解析命令行参数的艺术

&#x1f680; 掌握Perl命令行&#xff1a;深入解析命令行参数的艺术 在Perl编程中&#xff0c;命令行参数是与外部交互的重要方式之一。无论是执行脚本时的选项设置&#xff0c;还是传递必要的运行时数据&#xff0c;命令行参数都扮演着至关重要的角色。本文将带您深入了解如…

不容错过!手把手教你开启微信通话自动录音功能!(含手机端和电脑端)

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 微信自动录音 📒📝 方法一📝 方法二📝 电脑端自动录音📝 注意事项⚓️ 相关链接 ⚓️📖 介绍 📖 在商务沟通或重要对话中,通话录音功能可以帮助我们记录关键信息,避免遗漏,同时也是证据保存的一种手段。虽然微…

IPXProxy海外代理IP在MultiLogin指纹浏览器中的配置教程

Multilogin指纹浏览器是一款付费浏览器&#xff0c;它为用户提供了拥有多个虚拟浏览器配置文件的机会。作为最好的指纹浏览器之一&#xff0c;它常常被用来创建或管理多个账户&#xff0c;当然在这个过程&#xff0c;代理IP是不可或缺的一部分。下面给大家代理在MultiLogin指纹…

api文字识别智能录入、身份证识别、接口识别​

OCR技术和由此带来的文字识别自动化程度不断增加&#xff0c;不少人预计该技术将对相当一部分的行业、工作产生影响&#xff0c;其中有一部分是颠覆性的。比如文字录入的工作&#xff0c;现在不少企业为自己的系统、产品集成了OCR技术核心&#xff0c;不仅能够减少人工录入的压…

AI能耗短期不会造成电力短缺,算力能效长期改进空间巨大

人工智能&#xff08;AI&#xff09;的快速发展正引发其对能源消耗的普遍担忧。国际能源署&#xff08;IEA&#xff09;在2024年的报告中预测&#xff0c;由于AI和加密货币的增长&#xff0c;全球数据中心的用电量将在未来几年内翻倍。2022年&#xff0c;全球数据中心的用电量约…

WPF界面设计-更改按钮样式 自定义字体图标

一、下载图标文件 iconfont-阿里巴巴矢量图标库 二、xaml界面代码编辑 文件结构 &#xe653; 对应的图标代码 Fonts/#iconfont 对应文件位置 <Window.Resources><ControlTemplate TargetType"Button" x:Key"CloseButtonTemplate"…

将Hyper-V虚拟机与主机共享网络

Hyper-V 网络设置 目标 将Hyper-V虚拟机网络配置为与主机使用同一网络&#xff0c;并确保主机网络连接不受影响。 前提条件 主机上已安装Hyper-V已创建Hyper-V虚拟机 步骤 1. 配置主机网络共享 打开 控制面板 -> 网络和 Internet -> 网络连接。右键点击 WIAN,选择…

IMX6ULL linux4.x RS485配置

文章目录 IMX6ULL linux4.x RS485配置使用IMX6ULL硬件流控设备树 使用普通IO口做软件流控串口驱动补丁设备树 rs485测试程序使用效果 IMX6ULL linux4.x RS485配置 使用IMX6ULL硬件流控 设备树 pinctrl_485r1: 485r1grp {fsl,pins <MX6UL_PAD_UART2_TX_DATA__UART2_DCE_T…

【鸿蒙学习笔记】UIAbility组件概述

官方文档&#xff1a;UIAbility组件 目录标题 UIAbility组件概述 [Q&A] 什么是UIAbility&#xff1f;声明周期UIAbility组件-启动模式UIAbility组件-与UI的数据同步 UIAbility组件概述 [Q&A] 什么是UIAbility&#xff1f; UIAbility组件是一种包含UI界面的应用组件&a…

防火墙组网

一、实验拓扑图 二、实验要求 1、DMZ区内的服务器&#xff0c;办公区仅能在办公时间内&#xff08;9&#xff1a;00-18&#xff1a;00&#xff09;可以访问&#xff0c; 生产区的设备全天可以访问。 2、生产区不允许访问互联网&#xff0c;办公区和游客区允许访问互联网。 3、办…

PIP 换源:提升 Python 包安装速度的秘诀

一、引言 在使用 Python 进行开发时&#xff0c;我们经常需要通过 pip 命令安装各种库和依赖。然而&#xff0c;默认的源可能会因为网络原因导致下载速度缓慢&#xff0c;影响开发效率。这时候&#xff0c;换源就成为了一个非常实用的技巧。 二、为什么要换源&#xff1f; 提…

等保测评的创新与学习

等保测评的创新与学习 等保测评&#xff0c;即信息安全等级保护测评&#xff0c;是中国网络安全保障体系的核心组成部分。随着技术的发展和网络安全威胁的日益复杂&#xff0c;等保测评也在不断创新和学习&#xff0c;以适应新的安全需求。 创新实践 智能化测评工具的应用&…

Sora模型:释放创意产业文本到视频AI的潜力

Sora&#xff0c;这个由OpenAI在2024年推出的文本到视频生成模型&#xff0c;不仅能够将文字描述转化为生动的视频内容&#xff0c;而且还能保持视频一分钟之久的连贯性和高质量&#xff0c;这在之前是难以想象的。 尽管AI在图像和文本理解上已取得巨大进步&#xff0c;但将这…

python执行shell并获取结果

在Python中执行Shell命令并获取其结果&#xff0c;通常可以使用subprocess模块。这个模块允许我们启动新的进程&#xff0c;连接到它们的输入/输出/错误管道&#xff0c;并获取它们的返回码。下面是一个详细的示例&#xff0c;展示了如何使用subprocess.run()函数来执行Shell命…

飞猪惹怒12306,一张火车票让第三方平台耍尽手段……

小柴已经记不清铁路12306是多少次发出提醒&#xff0c;似乎每一次出行高峰&#xff0c;都会提醒一次。 比如一再强调&#xff0c;购买加速包、付费成为会员就能优先出票&#xff0c;找朋友助力砍一刀&#xff0c;就能获得更高的出票概率……都是假的。‍‍ 因为&#xff0c;铁…

GDB使用方法与命令介绍

GDB是一个调试器&#xff0c;可以允许你在程序运行的时候检查里面到底发生了什么 GDB使用方法&#xff1a; 首先编译可执行程序的时候需要加上-g参数&#xff0c;例如 gcc -g test.c -o test #编译时生成debug有关的程序信随后进入调试 gdb test//相关命令 r //全速运行//r…

Win-ARM联盟的端侧AI技术分析

Win-ARM联盟&#xff0c;端侧AI大幕将起 微软震撼发布全球首款AI定制Windows PC——Copilot PC&#xff0c;搭载全新NPU与重塑的Windows 11系统&#xff0c;纳德拉盛赞其为史上最快、最强、最智能的Windows PC。该设备算力需求高达40TOPS&#xff0c;支持语音翻译、实时绘画、文…

PHP同城多商户多行业系统小程序源码

同城新生态&#xff0c;解锁多商户多行业系统的无限魅力&#x1f306;&#x1f680; &#x1f308; 开篇&#xff1a;同城新纪元&#xff0c;多商户多行业系统引领潮流&#xff01; 想象一下&#xff0c;在同一个城市内&#xff0c;无论是美食探索、购物狂欢&#xff0c;还是…

滥用云服务进行传播的恶意软件越来越多

由于云服务提供了传统方式所不具备的可扩展性、匿名性和容错性&#xff0c;攻击者越来越多地开始利用云服务来存储、分发和建立 C&C 信道&#xff0c;例如 VCRUM 存储在 AWS 上或 SYK Cryptor 通过 DriveHQ 进行分发。 过去的一个月内&#xff0c;研究人员一直在监控使用这…