【Spring】 AOP面向切面编程

文章目录

  • AOP是什么?
  • 一、AOP术语名词介绍
  • 二、Spring AOP框架介绍和关系梳理
  • 三、Spring AOP基于注解方式实现和细节
    • 3.1 Spring AOP底层技术组成
    • 3.2 初步实现
    • 3.3 获取通知细节信息
    • 3.4 切点表达式语法
    • 3.5 重用(提取)切点表达式
    • 3.6 环绕通知
    • 3.7 切面优先级设置
    • 3.8 CGLib动态代理生效
    • 3.9 注解实现小结
  • 四、Spring AOP对获取Bean的影响理解
  • 总结
    • 代理介绍:
    • AOP名词理解:
    • AOP思维以及与代理的关系:


AOP是什么?

AOP:Aspect Oriented Programming面向切面编程,解决非核心代码的冗余。

  • AOP利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。
  • 所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。
  • 使用AOP,可以在不修改原来代码的基础上添加新功能。

AOP思想主要的应用场景:

  1. 日志记录:在系统中记录日志是非常重要的,可以使用AOP来实现日志记录的功能,可以在方法执行前、执行后异常抛出时记录日志
  2. 事务处理:在数据库操作中使用事务可以保证数据的一致性,可以使用AOP来实现事务处理的功能,可以在方法开始前开启事务,在方法执行完毕后提交或回滚事务
  3. 安全控制:在系统中包含某些需要安全控制的操作,如登录、修改密码、授权等,可以使用AOP来实现安全控制的功能。可以在方法执行前进行权限判断,如果用户没有权限,则抛出异常或转向到错误页面,以防止未经授权的访问。
  4. 性能监控:在系统运行过程中,有时需要对某些方法的性能进行监控,以找到系统的瓶颈并进行优化。可以使用AOP来实现性能监控的功能,可以在方法执行前记录时间戳,在方法执行完毕后计算方法执行时间并输出到日志中。
  5. 异常处理:系统中可能出现各种异常情况,如空指针异常、数据库连接异常等,可以使用AOP来实现异常处理的功能,在方法执行过程中,如果出现异常,则进行异常处理(如记录日志、发送邮件等)。
  6. 缓存控制:在系统中有些数据可以缓存起来以提高访问速度,可以使用AOP来实现缓存控制的功能,可以在方法执行前查询缓存中是否有数据,如果有则返回,否则执行方法并将方法返回值存入缓存中。
  7. 动态代理:AOP的实现方式之一是通过动态代理,可以代理某个类的所有方法,用于实现各种功能。
  • 综上所述,AOP可以应用于各种场景,它的作用是将通用的横切关注点与业务逻辑分离,使得代码更加清晰、简洁、易于维护。

一、AOP术语名词介绍

  • 横切关注点
    • 从每个方法中抽取出来的同一类非核心业务
    • 在同一个项目中,我们可以使用多个横切关注点对相关方法进行多个不同方面的增强。
    • 这个概念不是语法层面天然存在的,而是根据附加功能的逻辑上的需要:有十个附加功能,就有十个横切关注点

·

  • AOP把软件系统分为两个部分:核心关注点横切关注点

    • 业务处理的主要流程是核心关注点,非核心代码是横切关注点。
    • 横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事务、异常等。
    • AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。(分离核心业务代码与非核心代码
  • 通知(增强)

    • 每一个横切关注点上要做的事情都需要写一个方法来实现,这样的方法就叫通知方法。
      • 前置通知:在被代理的目标方法执行
      • 返回通知:在被代理的目标方法成功结束后执行(寿终正寝
      • 异常通知:在被代理的目标方法异常结束后执行(死于非命
      • 后置通知:在被代理的目标方法最终结束后执行(盖棺定论
      • 环绕通知:使用try...catch...finally结构围绕整个被代理的目标方法,包括上面四种通知对应的所有位置

1

  • 切入点 pointcut
    定位连接点的方式,或者可以理解成被选中的连接点!
    是一个表达式,比如execution(* com.spring.service.impl..(…))。
    符合条件的每个方法都是一个具体的连接点。

  • 切面 aspect
    切入点和通知的结合。是一个类。
    1

  • 目标 target
    被代理的目标对象。

  • 代理 proxy
    向目标对象应用通知之后创建的代理对象。

  • 织入 weave
    指把通知应用到目标上,生成代理对象的过程。
    可以在编译期织入,也可以在运行期织入,Spring采用后者。

二、Spring AOP框架介绍和关系梳理

  • AOP一种区别于OOP编程思维,用来完善和解决OOP的非核心代码冗余和不方便统一维护问题!
  • 代理技术(动态代理|静态代理)是实现AOP思维编程的具体技术,但是自己使用动态代理实现代码比较繁琐!
  • Spring AOP框架基于AOP编程思维封装动态代理技术,简化动态代理技术实现的框架!
    • SpringAOP内部帮助我们实现动态代理,
    • 我们只需写少量的配置指定生效范围即可完成面向切面思维编程的实现!

三、Spring AOP基于注解方式实现和细节

3.1 Spring AOP底层技术组成

1

  • 动态代理(InvocationHandler):JDK原生的实现方式,需要被代理的目标类必须实现接口
    • 因为这个技术要求代理对象和目标对象实现同样的接口(兄弟两个拜把子模式)。
  • cglib:通过继承被代理的目标类(认干爹模式)实现代理,所以不需要目标类实现接口。
  • AspectJ:早期的AOP实现的框架,SpringAOP借用了AspectJ中的AOP注解。

3.2 初步实现

1
1

  1. 导入依赖
<!-- spring-aspects会帮我们传递过来aspectjweaver -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>6.0.6</version>
</dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-aspects</artifactId><version>6.0.6</version>
</dependency>
  1. 准备接口
public interface Calculator {int add(int i, int j);int sub(int i, int j);int mul(int i, int j);int div(int i, int j);}
  1. 纯净实现类
/*** @Description: 实现计算接口,单纯添加加减乘除 实现,掺杂其它功能*  AOP -》 只针对IOC容器对象 - 创建代理对象 -> 将代理对象存储到IOC容器*/
@Component
public class CalculatorImpl implements Calculator {@Overridepublic int add(int i, int j) {int result = i + j;return result;}@Overridepublic int sub(int i, int j) {int result = i - j;return result;}@Overridepublic int mul(int i, int j) {int result = i * j;return result;}@Overridepublic int div(int i, int j) {int result = i / j;return result;}
}
  1. 声明切面类
package com.doug.advice;import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component;/*** @Description: 增强类 实现增强方法*/
@Component
@Aspect // @Aspect表示这个类是一个切面类
public class LogAspect {/** 1. 定义方法存储增强代码*       具体定义几个方法,根据插入的位置决定!* 2. 使用配置注解 指定插入目标方法的位置*   前置 @Before*   后置 @AfterReturning*   异常 @AfterThrowing*   最后 @After*   环绕 @Around**   try{*       前置*       目标方法执行*       后置*   }catch(){*       异常*   }finally{*       最后*   }** 3.配置切点表达式 [选择要插入的方法  切点* 4.注解补全*   加入IOC容器 @Component*   配置切面 @Aspect* 5.开启Aspect注解的支持* */@Before("execution(* com.doug.aop.impl.*.*(..))")public void start(){System.out.println("方法开始了");}@After("execution(* com.doug.aop.impl.*.*(..))")public void after(){System.out.println("方法结束了");}@AfterThrowing("execution(* com.doug.aop.impl.*.*(..))")public void error(){System.out.println("方法报错了");}
}
  1. 配置类
@Configuration
@ComponentScan("com.doug")
@EnableAspectJAutoProxy // 开启Aspectj的注解  xml: <aop:aspectj-autoproxy />
public class MyConfiguration {
}
  1. 测试
@SpringJUnitConfig(value = MyConfiguration.class)
public class TestAop {@Autowiredprivate Calculator calculator;@Testpublic void aopTest(){int add = calculator.add(1, 0);System.out.println(add);}
}

1

3.3 获取通知细节信息

  • JointPoint接口
    • JoinPoint 接口通过 getSignature() 方法获取目标方法的签名(方法声明时的完整信息)
  • 方法返回值
    • 在返回通知中,通过 @AfterReturning注解的returning属性获取目标方法的返回值!
  • 异常对象捕捉
    • 在异常通知中,通过**@AfterThrowing**注解的throwing属性获取目标方法抛出的异常对象
package com.doug.advice;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;import java.lang.reflect.Modifier;/*** @Description: 定义四个增强方法,获取目标方法的信息 返回值 异常*  1. 定义方法 - 增强代码*  2. 使用注解指定对应的位置*  3. 配置切点表达式选中方法*  4. 切面和IOC的配置*  5. 开启aspectj注解的支持*/
@Component
@Aspect
public class MyAdvice {/**   增强方法中获取目标方法信息:**  1. 全部增强方法中,获取目标方法的信息(方法名,参数,访问修饰符,所属类的信息...)*   (JoinPoint joinPoint) import org.aspectj.lang.JoinPoint;**  2. 返回结果 - @AfterReturning*     (Object result) result 接受返回的结果*       @AfterReturning(value = "execution(* com..impl.*.*(..))",returning = "result")** 3.  异常信息 - @AfterThrowing*   (Throwing t) t接收异常信息*   @AfterThrowing(value = "execution(* com..impl.*.*(..))",throwing = "result")* */@Before("execution(* com..impl.*.*(..))")public void start(JoinPoint joinPoint){// 1. 获取方法属于类的信息String simpleName = joinPoint.getTarget().getClass().getSimpleName();// 2. 获取方法名称int modifiers = joinPoint.getSignature().getModifiers();String s = Modifier.toString(modifiers);String name = joinPoint.getSignature().getName();//获取方法名// 3. 获取参数列表Object[] args = joinPoint.getArgs();}@AfterReturning(value = "execution(* com..impl.*.*(..))",returning = "result")public void afterReturn(JoinPoint joinPoint,Object result){}@After("execution(* com..impl.*.*(..))")public void after(){}@AfterThrowing(value = "execution(* com..impl.*.*(..))",throwing = "result")public void afterThrowing(Throwable result){}
}

3.4 切点表达式语法

切点表达式作用
AOP切点表达式(Pointcut Expression)是一种用于指定切点的语言,它可以通过定义匹配规则,来选择需要被切入的目标对象。
1
切点表达式语法
1
语法细节:
1

  • 实战:
1.查询某包某类下,访问修饰符是公有,返回值是int的全部方法execution(public int com.doug.xClass.*(..))
2.查询某包下类中第一个参数是String的方法execution(* com.doug.xClass.*(String..))
3.查询全部包下,无参数的方法!execution(* *..*.*())
4.查询com包下,以int参数类型结尾的方法execution(* com..*.*(..int))
5.查询指定包下,Service开头类的私有返回值int的无参数方法execution(private int com.doug.Service*.*())

3.5 重用(提取)切点表达式

1
提取重复的切点表达式
1
切点统一管理:
将切点表达式统一存储到一个类中进行集中管理和维护!

3.6 环绕通知

三合一(四合一)
1

3.7 切面优先级设置

相同目标方法上同时存在多个切面时,切面的优先级控制切面的内外嵌套顺序。

  • 优先级高的切面:外面
  • 优先级低的切面:里面

使用 @Order 注解可以控制切面的优先级:

  • @Order(较小的数):优先级高
  • @Order(较大的数):优先级低
    1
    实际意义:
    实际开发时,如果有多个切面嵌套的情况,要慎重考虑。例如:如果事务切面优先级高,那么在缓存中命中数据的情况下,事务切面的操作都浪费了。
    1
    此时应该将缓存切面的优先级提高,在事务操作之前先检查缓存中是否存在目标数据。
    1

3.8 CGLib动态代理生效

在目标类没有实现任何接口的情况下,Spring会自动使用cglib技术实现代理。

总结:
a. 如果目标类有接口,选择使用jdk动态代理

b. 如果目标类没有接口,选择cglib动态代理

c. 如果有接口,接口接值

d. 如果没有接口,类进行接值

3.9 注解实现小结

1

四、Spring AOP对获取Bean的影响理解

对实现了接口的类应用切面
在这里插入图片描述
对没实现接口的类应用切面new
1
如果使用AOP技术,目标类有接口,必须使用接口类型接收IoC容器中代理组件!


总结

代理介绍:

1

AOP名词理解:

1

AOP思维以及与代理的关系:

1

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

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

相关文章

一分钟 由浅入深 学会Navigation

目录 1.官网正式概念 1.1 初认知 2.导入依赖 2.1 使用navigation 2.2 safe Args插件-> 传递数据时用 3.使用Navigation 3.1 搭建初始框架 3.2 确定action箭头的属性 3.3 为Activity添加NavHostFragment控件 3.4 NavController 管理应用导航的对象 3.5 数据传递(单…

给定一个边与边可能相交的多边形,求它的轮廓线

大家好&#xff0c;我是前端西瓜哥。 最近遇到一个需求&#xff0c;给定一个多边形&#xff08;边与边可能相交&#xff09;&#xff0c;求这个多边形的轮廓线。 需要注意的是&#xff0c;轮廓线多边形内不能有空洞&#xff0c;使用的不是常见的非零绕数规则&#xff08;nonze…

SpringBoot3+Vue3 基础知识(持续更新中~)

bean 把方法的返回结果注入到ioc中 1: 2: 3: 组合注解封装 实战篇&#xff1a; 解析token&#xff1a; 统一携带token&#xff1a; 驼峰命名与下划线命名转换&#xff1a; NotEmpty!!! mybatis&#xff1a; PageHelper设置后&#xff0c;会将pageNum,和pageSize自己拼接…

代码随想录算法训练营第四一天 | 背包问题

目录 背包问题01背包二维dp数组01背包一维 dp 数组&#xff08;滚动数组&#xff09;分割等和子集 LeetCode 背包问题 01背包 有n件物品和一个最多能背重量为 w 的背包&#xff0c;第i件物品的重量是weight[i]&#xff0c;得到的价值是value[i] 。每件物品只能用一次&#x…

Python urllib、requests、HTMLParser

HTTP协议 HTTP 协议&#xff1a;一般指HTTP(超文本传输)协议。 HTTP是为Web浏览器和Web服务器之间的通信而设计的&#xff0c;基于TCP/IP通信协议嘞传递数据。 HTTP消息结构 客户端请求消息 客户端发送一个HTTP请求到服务器的请求消息包括以下格式 请求行(request line)请求…

【前端素材】推荐优质后台管理系统Start Admin平台模板(附源码)

一、需求分析 后台管理系统是一种用于管理网站、应用程序或系统的工具&#xff0c;它通常作为一个独立的后台界面存在&#xff0c;供管理员或特定用户使用。下面详细分析后台管理系统的定义和功能&#xff1a; 1. 定义 后台管理系统是一个用于管理和控制网站、应用程序或系统…

Linux——静态库

Linux——静态库 静态库分析一下 ar指令生成静态库静态库的使用第三方库优化一下 gcc -I(大写的i) -L -l(小写的l)&#xff0c;头文件搜索路径&#xff0c;库文件搜索路径&#xff0c;连接库 今天我们来学习静态库的基本知识。 静态库 在了解静态库之前&#xff0c;我们首先来…

复旦大学MBA聚劲联合会:洞见智慧,拓宽思维格局及国际化视野

12月2日&#xff0c;“焕拥时代 俱创未来”聚劲联合会俱创会年度盛典暨俱乐部募新仪式圆满收官。16家复旦MBA俱乐部、200余名同学、校友、各界同仁齐聚复旦管院&#xff0c;一起在精彩纷呈的圆桌论坛里激荡思想&#xff0c;在活力四射的俱乐部风采展示中凝聚力量。      以…

CSS 的圆角矩形

CSS 的圆角矩形 通过 border-radius 属性使矩形边框带圆角效果成为圆角矩形 语法&#xff1a;border-radius: length; length 是内切圆的半径&#xff0c;其数值越大, 弧线越明显 border-radius 属性值描述length定义圆角的形状%以百分比定义圆角的形状 生成圆形 让 border-…

高和汽车停工停产,创始人丁磊终于发话了!2024的冷门项目,投入小,但是真的很赚钱!

高和创始人丁磊站在停产停工的工厂呢&#xff0c; 环顾冷清❄️的四周&#xff0c;眉头紧锁&#x1f623;&#xff0c; 停顿片刻后对旁边同样愁眉苦脸的员工说道&#xff1a; 非常抱歉&#xff0c;因为经营的失误&#xff0c;面临了停产停工的窘境。 在互联网&#x1f517;、物…

C/C++的内存管理(2)——new与delete的内核与本质

内存管理 operator new 与 operator delete函数回看new与delete的实现内置类型自定义类型 常见面试题 我们已经知道了new与delete的用法及其好处&#xff0c;发现它似乎与C语言中的动态内存开辟的函数&#xff08;malloc/calloc/realloc&#xff09;不同 在这里我们特别指出&am…

二进制部署k8s集群之cni网络插件

目录 k8s的三种网络模式 pod内容器之间的通信 同一个node节点中pod之间通信 不同的node节点的pod之间通信 flannel网络插件 flannel的三种工作方式 VxLAN host-GW UDP Flannel udp 模式 Flannel VXLAN 模式 flannel插件的三大模式的总结 calico网络插件 k8s 组网…

命令绕过 [安洵杯 2019]easy_web1

打开题目 打开题目在URL处看到cmd&#xff0c;本能的直接用系统命令ls 发现被过滤了。又注意到imgTXpVek5UTTFNbVUzTURabE5qYz0似乎是一串base64 拿去base64解码 再hex解码一次得到555.png 再将其hex加密 base64加密 反向推出index.php的payload:?imgTmprMlJUWTBOalUzT0RK…

通过Colab部署Google最新发布的Gemma模型

Gemma的简单介绍 Gemma 是一系列轻量级、最先进的开放式模型&#xff0c;采用与创建 Gemini 模型相同的研究和技术而构建。 Gemma 由 Google DeepMind 和 Google 的其他团队开发&#xff0c;其灵感来自 Gemini&#xff0c;其名称反映了拉丁语 gemma&#xff0c;意思是“宝石”…

Promise相关理解记录

一、Promise基础定义相关 Promise是一个构造函数&#xff0c;调用时需要使用new关键字 Promise是解决回调地狱的一种异步解决方式 Promise有三个状态&#xff1a;pending(进行中)、fulfilled(成功)、rejected(失败) Promise的状态只会从 pending→fulfilled 或者 pending→…

Frida javascript hook 检测设备信息获取等

对 Android 应用进行 hook 常见的有 Xposed、Frida 等&#xff0c;Xposed 有时候可能不尽人意&#xff0c;或许您可以试试 Frida ~ frida -U -f com.primer.gamecerter -l hookStartActivity.js TODO 后续是否可以对检测数据&#xff08;堆栈、类名、方法名、参数、返回值&…

如何控制负压电源芯片的EN

上文我们探讨了如何将负压控制信号转变成正压&#xff0c;这样的信号通常是由负压的芯片产生的&#xff0c;比如负电压的电源管理芯片的power good信号&#xff0c;那么负压芯片应该由谁来控制呢&#xff1f;如何实现对负压电源管理芯片的有效控制呢&#xff1f;   举例&…

Yolov9全文翻译!

Yolo v9全文翻译 论文链接&#xff1a;&#x1f47f; YOLOv9: Learning What You Want to Learn Using Programmable Gradient Information 代码链接&#xff1a;&#x1f47f; https://github.com/WongKinYiu/yolov9/tree/main 大量图片来袭&#xff01;

Linux课程三课---Linux开发环境的使用(yum的相关)

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…

2023全新UI千月影视APP源码 | 前后端完美匹配、后端基于ThinkPHP框架

应用介绍 本文来自&#xff1a;2023全新UI千月影视APP源码 | 前后端完美匹配、后端基于ThinkPHP框架 - 源码1688 简介&#xff1a; 2023全新UI千月影视APP源码 | 前后端完美匹配、后端基于thinkphp框架 图片&#xff1a;