深入理解Spring AOP的工作流程

文章目录

    • 引言
    • 什么是AOP?
    • Spring AOP的工作原理
      • 1. JDK动态代理
      • 2. CGLIB代理
    • Spring AOP的注解方式
      • @Aspect注解
      • @EnableAspectJAutoProxy注解
    • Spring AOP的工作流程
    • 拓展应用
      • 1. 自定义注解
      • 2. 异常处理
      • 3. 切面优先级
    • 结论

在这里插入图片描述

🎉深入理解Spring AOP的工作流程


  • ☆* o(≧▽≦)o *☆嗨~我是IT·陈寒🍹
  • ✨博客主页:IT·陈寒的博客
  • 🎈该系列文章专栏:架构设计
  • 📜其他专栏:Java学习路线 Java面试技巧 Java实战项目 AIGC人工智能 数据结构学习
  • 🍹文章作者技术和水平有限,如果文中出现错误,希望大家能指正🙏
  • 📜 欢迎大家关注! ❤️

引言

在现代的软件开发中,面向切面编程(AOP)是一种重要的编程范式,用于解耦业务逻辑和横切关注点(cross-cutting concerns)。Spring框架提供了强大而灵活的AOP支持,通过代理机制实现横切关注点的注入。本文将深入探讨Spring AOP的工作流程,帮助读者更好地理解其原理和应用。

什么是AOP?

AOP是一种编程思想,通过在程序中间插入横切关注点,将系统划分为核心业务逻辑和横切关注点两部分。横切关注点包括日志记录、事务管理、安全控制等与核心业务逻辑无关但又必须在程序中执行的功能。AOP通过将这些横切关注点与核心业务逻辑分离,提高了代码的模块化和可维护性。

Spring AOP通过代理机制实现横切关注点的注入,其中代理对象负责执行横切逻辑。在Spring AOP中,常见的横切关注点包括日志记录、性能监控、事务管理等。

Spring AOP的工作原理

Spring AOP基于代理模式,主要通过两种方式实现:

  1. JDK动态代理: 基于接口的代理机制,使用java.lang.reflect.Proxy类生成代理对象。

  2. CGLIB代理: 基于类的代理机制,使用CGLIB库生成代理对象。

1. JDK动态代理

JDK动态代理要求目标类实现一个或多个接口,代理对象实现这些接口并委托给目标对象。以下是一个简单的例子:

public interface UserService {void addUser(String username, String password);
}public class UserServiceImpl implements UserService {@Overridepublic void addUser(String username, String password) {// 实际业务逻辑System.out.println("User added: " + username);}
}public class LogAspect implements InvocationHandler {private Object target;public LogAspect(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("Log: Method " + method.getName() + " is invoked");Object result = method.invoke(target, args);return result;}
}// 使用代理
public class Main {public static void main(String[] args) {UserService target = new UserServiceImpl();InvocationHandler handler = new LogAspect(target);UserService proxy = (UserService) Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),handler);proxy.addUser("John", "123456");}
}

2. CGLIB代理

CGLIB代理不要求目标类实现接口,代理对象继承目标对象。以下是一个简单的例子:

public class UserService {public void addUser(String username, String password) {// 实际业务逻辑System.out.println("User added: " + username);}
}public class LogAspect implements MethodInterceptor {@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("Log: Method " + method.getName() + " is invoked");Object result = proxy.invokeSuper(obj, args);return result;}
}// 使用代理
public class Main {public static void main(String[] args) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(UserService.class);enhancer.setCallback(new LogAspect());UserService proxy = (UserService) enhancer.create();proxy.addUser("John", "123456");}
}

Spring AOP的注解方式

除了基于XML的配置方式,Spring AOP还支持使用注解的方式配置切面。

@Aspect注解

在使用注解方式配置AOP时,首先需要使用@Aspect注解声明一个切面类。该类包含了多个切点和通知,用于定义横切逻辑。

@Aspect
@Component
public class LogAspect {@Pointcut("execution(* com.example.service.*.*(..))")public void pointcut() {}@Before("pointcut()")public void beforeAdvice(JoinPoint joinPoint) {System.out.println("Log: Before " + joinPoint.getSignature().getName());}@After("pointcut()")public void afterAdvice(JoinPoint joinPoint) {System.out.println("Log: After " + joinPoint.getSignature().getName());}
}

上述例子中,@Pointcut注解定义了一个切点,表示匹配com.example.service包下所有类的所有方法。@Before@After注解分别表示前置通知和后置通知。

@EnableAspectJAutoProxy注解

在Spring Boot中,还需要在配置类上使用@EnableAspectJAutoProxy注解开启自动代理功能。该注解告诉Spring Boot启用AspectJ自动代理。

@SpringBootApplication
@EnableAspectJAutoProxy
public class MyApplication {public static void main(String[] args) {SpringApplication.run(MyApplication.class, args);}
}

Spring AOP的工作流程

Spring AOP的工作流程可以概括为以下几个步骤:

  1. 定义切面: 创建一个Java类,并在类上使用@Aspect注解声明为切面类。在切面类中定义切点和通知。

  2. 配置通知: 使用@Before@After等注解配置通知,定义横切逻辑。

  3. 激活切面: 在配置类上使用@EnableAspectJAutoProxy注解激活切面。

  4. 容器初始化: Spring容器启动时,会扫描并解析所有标有@Aspect注解的类。

  5. 生成代理对象: 对于被代理的目标对象,Spring会根据切面定义生成代理对象。代理对象包含了横切逻辑。

  6. 执行横切逻辑: 在目标方法执行前、后或异常时,执行横切逻辑。

拓展应用

Spring AOP的应用远不止上述简单例子所示,还可以结合更复杂的切面和通知,实现更丰富的横切逻辑。以下是一些拓展应用的示例:

1. 自定义注解

可以使用自定义注解来标记切点,让代码更具可读性。例如,定义一个@Log注解,标记需要记录日志的方法。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Log {String value() default "";
}

然后,在切面类中使用@Around注解拦截被@Log注解标记的方法。

@Aspect
@Component
public class LogAspect {@Around("@annotation(com.example.annotation.Log)")public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("Log: Method " + joinPoint.getSignature().getName() + " is invoked");Object result = joinPoint.proceed();return result;}
}

2. 异常处理

通过@AfterThrowing注解可以实现异常处理逻辑,记录异常信息或进行其他处理。

@AfterThrowing(pointcut = "execution(* com.example.service.*.*(..))", throwing = "ex")
public void afterThrowingAdvice(JoinPoint joinPoint, Exception ex) {System.out.println("Exception: " + ex.getMessage());
}

3. 切面优先级

通过@Order注解可以指定切面的优先级,数字越小,优先级越高。

@Aspect
@Component
@Order(1)
public class LogAspect {// ...
}

在这里插入图片描述

结论

Spring AOP是Spring框架中一个重要的组件,通过代理机制实现横切关注点的注入。本文深入介绍了Spring AOP的工作原理,包括基于JDK动态代理和CGLIB代理的实现方式,以及使用注解配置AOP的方法。通过理解Spring AOP的工作流程,我们能更好地应用和拓展AOP,提高代码的模块化和可维护性。希望本文能够帮助读者更深入地理解和应用Spring AOP。


🧸结尾 ❤️ 感谢您的支持和鼓励! 😊🙏
📜您可能感兴趣的内容:

  • 【Java面试技巧】Java面试八股文 - 掌握面试必备知识(目录篇)
  • 【Java学习路线】2023年完整版Java学习路线图
  • 【AIGC人工智能】Chat GPT是什么,初学者怎么使用Chat GPT,需要注意些什么
  • 【Java实战项目】SpringBoot+SSM实战:打造高效便捷的企业级Java外卖订购系统
  • 【数据结构学习】从零起步:学习数据结构的完整路径

在这里插入图片描述

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

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

相关文章

关于运行软件程序出现vcruntime140.dll丢失的修复教程-解决方案

vcruntime140.dll是Microsoft Visual C库文件的一部分,用于支持Windows操作系统上的应用程序。如果找不到或丢失了这个文件,可能会导致某些应用程序无法正常运行。下面是关于vcruntime140.dll丢失的5个修复方法,以及vcruntime140.dll文件属性…

Python基础教程之分支结构详解

文章目录 一、分支结构二、单分支结构三、双分支结构四、多分支结构五、嵌套分支结构六、三元表达式七、条件测试关于Python技术储备一、Python所有方向的学习路线二、Python基础学习视频三、精品Python学习书籍四、Python工具包项目源码合集①Python工具包②Python实战案例③P…

Elasticsearch基础优化

分片策略 分片和副本得设计为ES提供支付分布式和故障转移得特性,但不意味着分片和副本是可以无限分配, 而且索引得分片完成分配后由于索引得路由机制,不能重新修改分片数(副本数可以动态修改) 一个分片得底层为一个l…

python之pyqt专栏2-项目文件解析

项目结构 在上一篇文章python之pyqt专栏1-环境搭建,创建新的pyqt项目,下面我们来看一下这个项目下的文件。 从下面的文件结构图可以看到,该项目下有3个文件,untitled.ui,untitled.py 以及main.py。 QtDesigner可以UI界面的方式&am…

Feign接口请求返回异常 no suitable HttpMessageConvert found for response type

问题场景&#xff1a; 后端调用feign接口请求, 接口返回异常, no suitable HttpMessageConvert found for response type 问题描述 报错异常如下&#xff1a; //根据图片特征 去查询人员信息ResultVo<List> personVos ipbdFaceLibPersonApi.queryFacePersonByFeatur…

华为云之SFS弹性文件服务使用体验

华为云之SFS弹性文件服务使用体验 一、本次实践介绍1.1 实践环境简介1.2 本次实践目的 二、SFS弹性文件服务介绍2.1 SFS弹性文件服务简介2.2 SFS弹性文件服务特点 三、购买ECS弹性云服务器3.1 购买ECS弹性云服务器3.2 查看ECS弹性云服务器状态3.3 远程连接ECS3.4 检查操作系统版…

package.json 中的版本号,符号 ~ 与 ^ 的区别及其他符号的含义

"element-ui": "^2.15.1", 版本号&#xff1a; 2.15.1 对应 x.y.z z &#xff1a;表示一些小的bugfix, 更改z的号&#xff1b;(修复补丁-z)&#xff08;~符号&#xff09; y &#xff1a;表示一些大的版本更改&#xff0c;比如一些API的变化&#xff1b;(…

迭代器模式 (Iterator Pattern)

定义 迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;用于顺序访问集合对象的元素&#xff0c;而无需知道集合对象的底层表示。迭代器模式将遍历集合的责任从集合对象转移到迭代器对象上&#xff0c;这简化了集合接口和实现&#xff0c;…

计算机应用基础_错题集_基础知识---网络教育统考工作笔记006

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、基础知识部分错题集总结前言 计算机应用基础统考,错题集总结 一、基础知识部分 基础知识部分 2、微处理器芯片的位数即指______。 A.速度 B.字长 C.

Unity Android FireBase bugly报错查询

报错如下图&#xff0c;注意&#xff0c;标红的三处 使用的il2cpp和架构是arm64-v8a 那我们就可以根据这些去找对应的符号表&#xff0c;在unity安装目录下 Unity2020.3.33f1\Editor\Data\PlaybackEngines\AndroidPlayer\Variations\il2cpp\Release\Symbols\arm64-v8a 找到l…

Tomcat注册为服务后,如何配置Tomcat内存大小

前提条件&#xff1a;tomcat已经注册为服务。 1.winR,输入regedit打开注册表 2.找到Tomcat注册表路径&#xff1a; HKEY_LOCAL_MACHINE\SOFTWARE\Wow6432Node\Apache Software Foundation\Procrun 2.0\Tomcat80603.找到jvm内存配置路径&#xff1a; HKEY_LOCAL_MACHINE\SOFTW…

PTA-6-312 使用多态实现图书馆计算罚金功能

题目&#xff1a; 图书馆提供给读者借阅服务&#xff0c;包括借阅书籍及音像制品。如果借阅超时需要缴纳罚金。对于不同类型的书籍和音像制品罚款规则不同。 成人书籍:允许借阅的时间是21天&#xff0c;每超时1天&#xff0c;需要缴纳罚金2元;如果超时3天以上&#xff0c;超出的…

vue中 mixin用法

在Vue.js中&#xff0c;mixin是一种可以在多个组件之间共享Vue组件选项的灵活方式。mixin对象可以包含任何组件选项。当组件使用mixin时&#xff0c;所有mixin对象的选项将被“混合”到该组件的选项中。 使用mixin的一个主要优点是可以在多个组件之间重用和共享代码。这可以帮…

SSM大学生社团信息管理系统-99953,(免费领取源码)计算机毕业设计选题开题+程序定制+论文书写+答辩ppt书写 包售后 全流程

SSM大学生社团信息管理系统APP 摘 要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;高校当然也不能排除在外。大学生社团信息管理系统APP是以实际运用为开发背景&#xff0c…

CMake中常见的预定义变量

文章目录 CMake常见的预定义变量CMake variables官方文档 CMake常见的预定义变量 在 CMake 中&#xff0c;有一些常见的预定义变量&#xff0c;它们提供了有关项目、目录结构和构建环境的信息。这些变量可用于设置路径、传递参数、以及进行其他与构建过程相关的操作。 以下是…

Rust语言入门教程(五) - 流控制语句

if 表达式 在Rust中&#xff0c; if语句的判断条件不需要用( )括起来&#xff0c; 它会认为所有在if 和 {之间的表达式就是判断条件&#xff0c;例如&#xff1a; if num 5 {msg "five"; }判断条件的表达式必须返回一个bool型的值&#xff0c; 因为Rust是一个不喜…

[C/C++]数据结构 循环队列

前言: 队列是一种具有先进先出特性的结构,但是当数据出队列以后,前面的空间就无法再次利用了,循环队列就可以解决这个问题 一:概念及结构: 1.循环队列概念 循环队列是一种线性数据结构&#xff0c;其操作表现基于 FIFO&#xff08;先进先出&#xff09;原则并且队尾被连接在队…

颠覆与创新:算法备案的双重挑战

随着数字时代的迅猛发展&#xff0c;算法已经成为了企业创新和竞争的关键因素。然而&#xff0c;伴随着算法的广泛应用&#xff0c;数据隐私、法规合规等问题也愈发凸显&#xff0c;给企业带来了双重挑战。本文将深入探讨这一话题&#xff0c;探讨算法备案如何在颠覆与创新之间…

IDEA、PHPSTORM 在命令行中进行 PHP debug

然在终端执行控制器的方法php yii test/ab 即可看到触发debug 调试

视频剪辑技巧:多个视频合并新篇章,高效视频剪辑,创造无限可能

在数字媒体时代&#xff0c;视频剪辑已经成为一项重要的技能。多个视频合并是一种将多个视频片段合并成一个完整视频的技巧。这种技巧可以将不同的视频片段组合在一起&#xff0c;制作出独特且具有吸引力的视频内容。现在一起操作下云炫AI智剪如何批量合并视频的操作吧。 一、准…