百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

一文搞懂!Spring AOP原理!(什么是spring aop原理)

yuyutoo 2025-01-23 22:12 3 浏览 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使用ProxyFactoryAdvisedSupport等类来管理和创建代理对象。代理又可以细分为:

  • 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通知:在目标方法抛出异常后执行。

执行目标方法

在执行完BeforeAround通知的前置逻辑后,代理对象会调用目标对象的实际方法。目标方法执行完成后,代理对象会继续执行AfterAround的后置逻辑、AfterReturningAfterThrowing通知。

返回结果或抛出异常

代理对象在完成所有通知逻辑后,将目标方法的返回结果返回给调用方。如果目标方法抛出异常,代理对象也会处理异常并根据配置决定是否重新抛出或转换异常。

结束

AOP的执行流程在代理对象返回结果或抛出异常后结束,整个过程是透明的,调用方无需关心代理的存在,目标对象的行为在运行时被增强。

Spring AOP核心源码分析

Spring AOP的源码涉及到多个核心类和接口,包括ProxyFactoryAdvisedSupportAopProxyJdkDynamicAopProxyCglibAopProxy等。下面,我们将对这些核心组件进行详细分析。

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的强大之处在于其灵活性和可扩展性,使得开发者可以轻松地实现横切关注点的分离和复用。

相关推荐

微软Win10/Win11版Copilot上线:支持OpenAI o3推理模型

IT之家4月3日消息,科技媒体WindowsLatest昨日(4月2日)发布博文,报道称Windows10、Windows11新版Copilot应用已摘掉Beta帽...

WinForm 双屏幕应用开发:原理、实现与优化

在当今的软件开发领域,多屏幕显示技术的应用越来越广泛。对于WinForm应用程序来说,能够支持双屏幕显示不仅可以提升用户体验,还能满足一些特定场景下的业务需求,比如在演示、监控或者多任务处理等场景...

推荐一个使用 C# 开发的 Windows10 磁贴美化小工具

...

OpenJDK 8 安装(openjdk 8 windows)

通常OpenJDK8和11都能互相编译和通用。我们建议使用11,但是如果你使用JDK8的话也是没有问题的。建议配置使用OpenJDK,不建议使用OracleJDK,主要是因为版...

基于 Linux 快速部署 OpenConnect VPN 服务(ocserv 实战指南)

一、前言在如今远程办公和内网穿透需求日益增长的背景下,搭建一套安全、稳定、高效的VPN系统显得尤为重要。OpenConnectServer(ocserv)是一个开源、高性能的VPN服务端软件...

巧妙设置让Edge浏览器更好用(edge怎么设置好用)

虽然现在新版本的Edge浏览器已经推出,但是毕竟还处于测试的状态中。而Win10系统里面自带的老版Edge浏览器,却越来越不被人重视。其实我们只需要根据实际情况对老版本的Edge浏览器进行一些简单的设...

WPF做一个漂亮的登录界面(wpf页面设计)

...

微软开源博客工具Open Live Writer更新:多项Bug修复

OpenLiveWriter前身是WindowsLiveWriter,是微软WindowsLive系列软件之一,曾经是博主们非常喜爱的一款所见即所得博文编辑工具,支持离线保存,还支持图像编辑...

基于OpenVINO的在线设计和虚拟试穿 | OPENAIGC大赛企业组优秀作品

在第二届拯救者杯OPENAIGC开发者大赛中,涌现出一批技术突出、创意卓越的作品。为了让这些优秀项目被更多人看到,我们特意开设了优秀作品报道专栏,旨在展示其独特之处和开发者的精彩故事。...

C#开源免费的Windows右键菜单管理工具

...

Windows10或11中隐藏的功能,用它再也不用担心电脑中病毒!

...

Python open函数详解(python open函数源码)

演示环境,操作系统:Win1021H2(64bit);Python解释器:3.8.10。open是Python的一个内置函数,一般用于本地文件的读写操作。用法如下。my_file=open(fi...

Windows 11 安装 Docker Desktop(Windows 11 安装助手 Windows 易升 关系)

...

Windows 11 新版发布:屏幕亮度自适应控制,小组件界面重新设计!

...

世界上最好用的Linux发行版之一,OpenSUSE安装及简单体验

背景之前无意在论坛里看到openSUSE的Linux发行版,被称为世界上最好用的Linux发行版之一(阔怕),一直想体验一下,于是这期做一个安装和简单体验教程吧。...

取消回复欢迎 发表评论: