一文搞懂!Spring AOP原理!(什么是spring aop原理)
yuyutoo 2025-01-23 22:12 1 浏览 0 评论
Spring AOP是 Spring框架中的一个重要模块,它通过分离关注点来提高代码的模块化程度,AOP允许开发者在不改变业务逻辑的情况下,通过切面来增强或修改代码的行为。本文我们将深入分析 Spring AOP的原理。
Spring AOP概述
什么是AOP?
AOP,全程 Aspect-Oriented Programming,中文翻译为面向切面编程,它是一种编程范式,旨在通过将横切关注点(如日志记录、事务管理、权限控制等)分离出来,使得这些关注点可以独立于业务逻辑进行处理。AOP的核心概念包括:
- 切面(Aspect):模块化的关注点,通常横切多个对象。
- 连接点(Join Point):程序执行过程中的某个点,比如方法调用或异常抛出。
- 通知(Advice):在切面的某个特定的连接点上执行的动作。
- 切入点(Pointcut):匹配连接点的断言。
- 目标对象(Target Object):被通知的对象。
- 代理(Proxy):通知目标对象后,创建的对象。
- 织入(Weaving):将切面连接到其它应用程序类型或对象上,并创建一个通知对象。
Spring AOP的核心原理
AOP的实现机制
Spring AOP基于代理模式实现,主要通过Proxy对象来替代目标对象,并在Proxy对象的方法调用中插入切面逻辑。Spring AOP使用ProxyFactory和AdvisedSupport等类来管理和创建代理对象。代理又可以细分为:
- JDK动态代理:基于接口的代理,目标对象必须实现一个或多个接口。
- CGLIB代理:基于子类的代理,适用于目标对象没有实现接口的情况。
AOP的核心组件
- Advisor:包含切入点和通知的元数据。
- Advice:定义切面在连接点上执行的操作。
- Pointcut:定义匹配连接点的规则。
- AopProxy:负责创建代理实例,具体实现有JdkDynamicAopProxy和CglibAopProxy。
AOP的执行流程
Spring AOP的执行流程是理解其工作原理的关键,它通过在程序运行时动态地将切面逻辑织入到目标对象中,从而实现横切关注点的分离。下面我们来详细地分析 Spring AOP的执行流程。
配置切面
AOP的执行流程从配置切面开始,切面可以通过 XML配置文件或基于注解的方式进行定义。配置切面时,主要涉及以下几个元素:
- 切面类(Aspect):包含横切逻辑的类,通常用@Aspect注解标识。
- 通知方法(Advice):定义在特定连接点上执行的横切逻辑。通知类型包括@Before、@After、@Around、@AfterReturning、@AfterThrowing等。
- 切入点表达式(Pointcut Expression):用于匹配连接点的方法执行点,通常使用AspectJ的切入点表达式语法。
创建代理对象
在Spring容器启动时,Spring会扫描配置的切面类,并为每个目标对象创建代理对象。代理对象负责在目标方法执行前后插入切面逻辑。Spring AOP使用两种主要的代理方式:
- JDK动态代理:适用于目标对象实现了接口的情况,通过Java的反射机制创建代理对象。
- CGLIB代理:适用于目标对象没有实现接口的情况,通过生成目标类的子类来创建代理。
方法调用拦截
当客户端代码调用目标对象的方法时,实际上是通过代理对象来进行调用的。代理对象实现了与目标对象相同的接口,因此客户端代码无需感知代理的存在。
- 拦截方法调用:代理对象拦截对目标方法的调用。对于JDK动态代理,这是通过实现InvocationHandler接口的invoke方法来实现的;对于CGLIB代理,这是通过生成子类并重写方法来实现的。
执行通知
在方法调用被拦截后,代理对象会根据切面配置执行相应的通知逻辑:
- Before通知:在目标方法执行之前执行。
- After通知:在目标方法执行之后执行,无论方法是否抛出异常。
- Around通知:包围目标方法的执行,可以在方法执行前后进行自定义逻辑,甚至可以决定是否执行目标方法。
- AfterReturning通知:在目标方法成功返回后执行。
- AfterThrowing通知:在目标方法抛出异常后执行。
执行目标方法
在执行完Before或Around通知的前置逻辑后,代理对象会调用目标对象的实际方法。目标方法执行完成后,代理对象会继续执行After、Around的后置逻辑、AfterReturning或AfterThrowing通知。
返回结果或抛出异常
代理对象在完成所有通知逻辑后,将目标方法的返回结果返回给调用方。如果目标方法抛出异常,代理对象也会处理异常并根据配置决定是否重新抛出或转换异常。
结束
AOP的执行流程在代理对象返回结果或抛出异常后结束,整个过程是透明的,调用方无需关心代理的存在,目标对象的行为在运行时被增强。
Spring AOP核心源码分析
Spring AOP的源码涉及到多个核心类和接口,包括ProxyFactory、AdvisedSupport、AopProxy、JdkDynamicAopProxy、CglibAopProxy等。下面,我们将对这些核心组件进行详细分析。
ProxyFactory
ProxyFactory是Spring AOP创建代理的核心工厂类。它负责根据配置创建合适的代理对象(JDK动态代理或CGLIB代理)。
public class ProxyFactory extends ProxyCreatorSupport {
// 获取代理对象
public Object getProxy() {
return createAopProxy().getProxy();
}
// 创建AopProxy对象
protected AopProxy createAopProxy() {
if (!this.isProxyTargetClass()) { // 是否强制使用CGLIB代理
return new JdkDynamicAopProxy(this);
}
return new CglibAopProxy(this);
}
}
- getProxy():对外提供获取代理对象的方法。
- createAopProxy():根据ProxyTargetClass属性判断使用JDK动态代理还是CGLIB代理。
AdvisedSupport
AdvisedSupport是Spring AOP的配置类,持有AOP代理需要的各种配置,包括目标对象、切面、通知等。
public class AdvisedSupport extends ProxyConfig implements Advised {
private TargetSource targetSource;
private List<Advisor> advisors = new ArrayList<>();
private List<Class<?>> interfaces = new ArrayList<>();
// 其他配置和方法
}
- TargetSource:封装了目标对象。
- advisors:存储应用于目标对象的通知(Advice)和切入点(Pointcut)。
- interfaces:代理对象需要实现的接口列表。
AopProxy接口
AopProxy是一个接口,定义了AOP代理对象的创建方法。
public interface AopProxy {
Object getProxy();
Object getProxy(ClassLoader classLoader);
}
- getProxy():用于创建代理对象。
- getProxy(ClassLoader classLoader):允许指定类加载器创建代理对象。
JdkDynamicAopProxy
JdkDynamicAopProxy实现了AopProxy接口,使用JDK动态代理为目标对象创建代理。
public class JdkDynamicAopProxy implements AopProxy, InvocationHandler {
private final AdvisedSupport advised;
public JdkDynamicAopProxy(AdvisedSupport config) {
this.advised = config;
}
@Override
public Object getProxy() {
return getProxy(Thread.currentThread().getContextClassLoader());
}
@Override
public Object getProxy(ClassLoader classLoader) {
return Proxy.newProxyInstance(classLoader, this.advised.getProxiedInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, this.advised.getTargetClass());
if (chain.isEmpty()) {
return method.invoke(this.advised.getTargetSource().getTarget(), args);
}
MethodInvocation invocation = new ReflectiveMethodInvocation(proxy, this.advised.getTargetSource().getTarget(), method, args, chain);
return invocation.proceed();
}
}
- getProxy():通过Proxy.newProxyInstance创建代理对象。
- invoke():实现InvocationHandler接口的方法,负责方法调用的拦截和通知链的执行。
CglibAopProxy
CglibAopProxy同样实现了AopProxy接口,使用CGLIB库为目标对象创建代理。
public class CglibAopProxy implements AopProxy {
private final AdvisedSupport advised;
public CglibAopProxy(AdvisedSupport config) {
this.advised = config;
}
@Override
public Object getProxy() {
return getProxy(Thread.currentThread().getContextClassLoader());
}
@Override
public Object getProxy(ClassLoader classLoader) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(this.advised.getTargetClass());
enhancer.setInterfaces(this.advised.getProxiedInterfaces());
enhancer.setCallback(new DynamicAdvisedInterceptor(this.advised));
return enhancer.create();
}
private static class DynamicAdvisedInterceptor implements MethodInterceptor {
private final AdvisedSupport advised;
public DynamicAdvisedInterceptor(AdvisedSupport advised) {
this.advised = advised;
}
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, this.advised.getTargetClass());
if (chain.isEmpty()) {
return proxy.invokeSuper(obj, args);
}
MethodInvocation invocation = new CglibMethodInvocation(obj, this.advised.getTargetSource().getTarget(), method, args, proxy, chain);
return invocation.proceed();
}
}
}
- getProxy():使用CGLIB的Enhancer创建代理对象。
- DynamicAdvisedInterceptor:CGLIB的拦截器实现,负责方法调用的拦截和通知链的执行。
MethodInvocation
MethodInvocation接口及其实现类(如ReflectiveMethodInvocation)负责封装方法调用的上下文信息,并管理通知链的执行。
public interface MethodInvocation extends Joinpoint {
Method getMethod();
Object[] getArguments();
}
public class ReflectiveMethodInvocation implements MethodInvocation {
private final Object proxy;
private final Object target;
private final Method method;
private final Object[] arguments;
private final List<?> interceptorsAndDynamicMethodMatchers;
private int currentInterceptorIndex = -1;
@Override
public Object proceed() throws Throwable {
if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
return this.method.invoke(this.target, this.arguments);
}
Object interceptorOrInterceptionAdvice = this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
if (interceptorOrInterceptionAdvice instanceof MethodInterceptor) {
MethodInterceptor interceptor = (MethodInterceptor) interceptorOrInterceptionAdvice;
return interceptor.invoke(this);
} else {
return proceed();
}
}
}
- proceed():递归调用通知链中的下一个拦截器,最终执行目标方法。
通过对 Spring AOP源码的详细分析,我们可以看到Spring AOP是如何通过代理模式实现面向切面编程的。
Spring AOP应用示例
下面我们通过一个简单的 Spring AOP示例,展示如何通过AOP实现日志记录。
定义业务类
public class UserService {
public void createUser(String username) {
System.out.println("Creating user: " + username);
}
}
定义切面
@Aspect
public class LoggingAspect {
@Before("execution(* UserService.createUser(..))")
public void logBefore(JoinPoint joinPoint) {
System.out.println("Before method: " + joinPoint.getSignature().getName());
}
}
Spring配置
使用Java配置:
@Configuration
@EnableAspectJAutoProxy
public class AppConfig {
@Bean
public UserService userService() {
return new UserService();
}
@Bean
public LoggingAspect loggingAspect() {
return new LoggingAspect();
}
}
测试AOP功能
public class AopTest {
public static void main(String[] args) {
ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
UserService userService = context.getBean(UserService.class);
userService.createUser("Alice");
}
}
输出结果:
Before method: createUser
Creating user: Alice
总结
Spring AOP通过代理模式实现了面向切面编程,能够在不改变业务逻辑的情况下增强代码功能。通过本文的分析,我们了解了 Spring AOP的基本概念、实现机制、核心组件以及如何在实际项目中应用 AOP。Spring AOP的强大之处在于其灵活性和可扩展性,使得开发者可以轻松地实现横切关注点的分离和复用。
相关推荐
- 史上最全的浏览器兼容性问题和解决方案
-
微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...
-
- 平面设计基础知识_平面设计基础知识实验收获与总结
-
CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...
-
2025-02-21 16:01 yuyutoo
- 写作排版简单三步就行-工具篇_作文排版模板
-
和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...
- 写一个2048的游戏_2048小游戏功能实现
-
1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...
- 今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……
-
最近的天气暖和得让很多小伙伴们喊“热”!!! 昨天的气温到底升得有多高呢?你家有没有榜上有名?...
- CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式
-
之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...
- 你的自我界限够强大吗?_你的自我界限够强大吗英文
-
我的结果:A、该设立新的界限...
- 行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?
-
行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...
-
- 让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
-
去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...
-
2025-02-21 16:00 yuyutoo
- 今年国家综合性消防救援队伍计划招录消防员15000名
-
记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...
- 一起盘点最新 Chrome v133 的5大主流特性 ?
-
1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...
- 竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使
-
style="text-align:center;"data-mce-style="text-align:...
- 学物理能做什么?_学物理能做什么 卢昌海
-
作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...
-
- 你不知道的关于这只眯眼兔的6个小秘密
-
在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...
-
2025-02-21 16:00 yuyutoo
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)