Spring AOP

简介: Spring AOP

一. 什么事AOP

AOP(Aspect Oriented Programming):面向切面编程,它是一种思想,它是对某一类事务的集中处理。比如用户登录权限的校验,再没学AOP之前,我们所有需要判断用户登录的界面(中的方法),都要各自实现或调用用户验证的方法,然而有了AOP之后,我们只需要再某一处配置一下,所有需要判断用户登录页面就都可以实现用户登录验证了,不需要每个方法都写相同的用户登录验证了。


而AOP 是一种思想,而Spring AOP 是一个框架,提供了一种对AOP 思想的实现,它们的关系和IoC 与 DI 类似

二. AOP 的组成

2.1 切面(Aspect)

切面由切点和通知组成,它既包含了横切逻辑的定义,也包含了连接点的定义

切面是包含了:通知,切点和切面的类,相当于 AOP 实现了某个功能的集合(定义一个事件:用户登录校验)

2.2 连接点(Join Point)

应用执行过程中能够插入切面的一个点,这个点可以是方法调用时,抛出异常,甚至修改字段时。切面代码可以利用这些点插入到应用的正常流程之中,并添加新的行为。

连接点相当于需要被增强的某个 AOP 功能的所有方法(有可能触发切点的所有点:所有接口)

2.3 切点(Pointcut)

Pointcut 的作用就是提供一组规则(使用 AspectJ pointcut expression language 来描述)来匹配 Join Point , 给满足规则的 Join Point 添加 Advice.


切点相当于保存了众多连接点的一个集合(定义用户登录拦截规则,哪些接口需要判断用户登录,哪些不判断)

2.4 通知(Advice)

AOP 执行的具体方法。定义了切面是什么,何时使用,其描述了切面要完成的工作。

Spring 切面类中,可以在方法上使用以下注解,会设置方法为通知方法,在满足条件后会通知本方法进行调用:

  • 前置通知:@Before: 通知方法会在目标方法调用之前执行
  • 后置通知:@After:通知方法会在目标方法返回或者抛出异常后调用
  • 返回通知:@AfterReturning:通知方法会在目标方法返回后调用
  • 异常通知:@AfterThrowing:通知方法会在目标方法抛出异常后调用
  • 环绕通知:@Around:通知包裹了被通知的方法,在被通知的方法通知之前和调用之后执行自定义的行为。

三. Spring AOP 实现

Spring AOP 的实现步骤如下:

3.1 添加Spring AOP 框架支持

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

3.2 定义切面和切点

//表明此类为一个切面
@Aspect
@Component
public class UserAspect {
    //定义一个切点,这里使用AspectJ 表达式语法
    @Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
    public void poingcut(){}
}

其中pointcut 为空方法,它不需要有方法体,此方法名就是起到一个“标识”的作用,标识下面的通知方法具体指的是哪个切点(因为切点可能会有很多个)

切点表达式支持三种通配符:

  • *:匹配任意字符,只匹配一个元素(包类,或方法,方法参数)
  • ..:匹配任意字符,可以匹配多个元素,在表示类时,必须和 * 联合使用
  • +:表示按照类型匹配指定类的所有类,必须跟在类名后面,如 com.cad.Car+,表示继承该类的所有子类包括本身

切点表达式由切点函数组成,其中 execution() 是最常用的切点函数,用来匹配方法,语法为:

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

修饰符和异常通常可以省略,具体含义如下:

修饰符,一般省略

public 公共方法
* 任意

返回值,不能省略

void 没有返回值
String 返回值字符串
* 任意

com.gyf.crm 固定包
com.gyf.crm.*.service crm包下面子包任意(ex:com.guf.crm.staff.service)
com.gyf.crm.. crm包下面任意子包(包括自己)
com.gyf.crm.*.service.. crm包下任意子包,固定目录service,service目录任意包

UerService 指定类
*lmpl 以 lmpl 结尾
User* 以 User 开头
* 任意

方法名,不能省略

addUser 固定方法
add* 以 add 开头
*Do 以 Do 结尾
* 任意


参数

() 无参
(int) 一个整型
(int,int) 两个整型
(..) 任意参数


表达式示例

execution(* com.cad.demo.User.*(..)) 匹配User 类里的所有方法
execution(* com.cad.demo.User+.*(..)) 匹配该类的子类包括该类的所有方法
execution(* com.cad.*.*(..)) 匹配 com.cad 包下所有类的所有方法
execution(* com.cad..*.*(..)) 匹配 com.cad 包下、子孙包下所有类的所有方法
execution(* addUser(String,int)) 匹配 addUser 方法,且第一个参数类型是 String,第二个参数类型是 int

3.4 定义通知

    //前置通知
    @Before("pointcut()")
    public void doBefore(){
        System.out.println("执行了 doBefore 方法");
    }
    //后置通知
    @After("pointcut()")
    public void doAfter(){
        System.out.println("执行了 doAfter 方法");
    }
    //返回之后通知
    @AfterReturning("pointcut()")
    public void doAfterReturning(){
        System.out.println("执行了 doAfterReturning 方法");
    }
    //抛出异常之后通知
    @AfterThrowing("pointcut()")
    public void doAfterThrowing(){
        System.out.println("执行了 doAfterThrowing 方法");
    }
    //环绕通知(传入的参数类型 ProceedingJoinPoint 是固定的)
    @Around("pointcut()")
    public Object doAround(ProceedingJoinPoint joinPoint){
        Object result=null;
        System.out.println("doAround 方法开始执行");
        try {
            result=joinPoint.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("doAround 方法结束执行");
        return result;
    }
    @RequestMapping("/login.html")
    public String login(HttpServletRequest request){
        HttpSession session=request.getSession();
        session.setAttribute("username","张三");
        System.out.println("登录成功");
        return "login";
    }

访问后的打印结果:

doAround 方法开始执行

执行了 doBefore 方法

登录成功

执行了 doAfterReturning 方法

执行了 doAfter 方法

doAround 方法结束执行

四. Spring AOP 实现原理

Spring AOP 是构建在动态代理基础上,因此 Spring 对 SOP 的支持局限于方法级别的拦截

Spring AOP 支持 JDK Proxy 和 CGLIB 方式实现动态代理.默认情况下,实现了接口的类,使用AOP 会基于JDK 生成代理类,没有实现接口的类,会基于 CGLIB 生成代理类.二者都是基于反射实现的。

JDK 动态代理实现:

import org.example.demo.service.AliPayService;
        import org.example.demo.service.PayService;
        import java.lang.reflect.InvocationHandler;
        import java.lang.reflect.Method;
        import java.lang.reflect.Proxy;
//动态代理:使⽤JDK提供的api(InvocationHandler、Proxy实现),此种⽅式实现,要求被代理类必须实现接⼝
public class PayServiceJDKInvocationHandler implements InvocationHandler {
    //⽬标对象即就是被代理对象
    private Object target;
    public PayServiceJDKInvocationHandler( Object target) {
        this.target = target;
    }
    //proxy代理对象
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//1.安全检查
        System.out.println("安全检查");
//2.记录⽇志
        System.out.println("记录⽇志");
//3.时间统计开始
        System.out.println("记录开始时间");
//通过反射调⽤被代理类的⽅法
        Object retVal = method.invoke(target, args);
//4.时间统计结束
        System.out.println("记录结束时间");
        return retVal;
    }
    public static void main(String[] args) {
        PayService target= new AliPayService();
//⽅法调⽤处理器
        InvocationHandler handler =
                new PayServiceJDKInvocationHandler(target);
//创建⼀个代理类:通过被代理类、被代理类实现的接⼝、⽅法调⽤处理器来创建
        PayService proxy = (PayService) Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                new Class[]{PayService.class},
                handler
        );
        proxy.pay();
    }
}

CGLIB 动态代理实现:

import org.springframework.cglib.proxy.Enhancer;
        import org.springframework.cglib.proxy.MethodInterceptor;
        import org.springframework.cglib.proxy.MethodProxy;
        import org.example.demo.service.AliPayService;
        import org.example.demo.service.PayService;
        import java.lang.reflect.Method;
public class PayServiceCGLIBInterceptor implements MethodInterceptor {
    //被代理对象
    private Object target;
    public PayServiceCGLIBInterceptor(Object target){
        this.target = target;
    }
    @Override
    public Object intercept(Object o, Method method, Object[] args, Method
            Proxy methodProxy) throws Throwable {
//1.安全检查
        System.out.println("安全检查");
//2.记录⽇志
        System.out.println("记录⽇志");
//3.时间统计开始
        System.out.println("记录开始时间");
//通过cglib的代理⽅法调⽤
        Object retVal = methodProxy.invoke(target, args);
//4.时间统计结束
        System.out.println("记录结束时间");
        return retVal;
    }
    public static void main(String[] args) {
        PayService target= new AliPayService();
        PayService proxy= (PayService) Enhancer.create(target.getClass(),
                new PayServiceCGLIBInterceptor(target));
        proxy.pay();
    }
}

区别:


  • 出身不同,JDK Proxy 是来自于原生的jdk,而 CGLIB 是第三方库的
  • JDK Proxy 要求被代理类必须实现接口,CGLIB 可以不实现接口,可以通过实现代理类的子类来实现动态代理
  • CGLIB 动态代理无法代理 final 类和 final 方法;JDK 动态代理可以代理任意类
  • JDK 动态代理性能相对较高,生成代理对象速度较快;CGLIB 动态代理性能相对较低,生成代理对象速度较慢。
相关文章
|
20天前
|
XML 安全 Java
使用 Spring 的 @Aspect 和 @Pointcut 注解简化面向方面的编程 (AOP)
面向方面编程(AOP)通过分离横切关注点,如日志、安全和事务,提升代码模块化与可维护性。Spring 提供了对 AOP 的强大支持,核心注解 `@Aspect` 和 `@Pointcut` 使得定义切面与切入点变得简洁直观。`@Aspect` 标记切面类,集中处理通用逻辑;`@Pointcut` 则通过表达式定义通知的应用位置,提高代码可读性与复用性。二者结合,使开发者能清晰划分业务逻辑与辅助功能,简化维护并提升系统灵活性。Spring AOP 借助代理机制实现运行时织入,与 Spring 容器无缝集成,支持依赖注入与声明式配置,是构建清晰、高内聚应用的理想选择。
252 0
|
4月前
|
监控 安全 Java
Spring AOP实现原理
本内容主要介绍了Spring AOP的核心概念、实现机制及代理生成流程。涵盖切面(Aspect)、连接点(Join Point)、通知(Advice)、切点(Pointcut)等关键概念,解析了JDK动态代理与CGLIB代理的原理及对比,并深入探讨了通知执行链路和责任链模式的应用。同时,详细分析了AspectJ注解驱动的AOP解析过程,包括切面识别、切点表达式匹配及通知适配为Advice的机制,帮助理解Spring AOP的工作原理与实现细节。
|
1月前
|
人工智能 监控 安全
Spring AOP切面编程颠覆传统!3大核心注解+5种通知类型,让业务代码纯净如初
本文介绍了AOP(面向切面编程)的基本概念、优势及其在Spring Boot中的使用。AOP作为OOP的补充,通过将横切关注点(如日志、安全、事务等)与业务逻辑分离,实现代码解耦,提升模块化程度、可维护性和灵活性。文章详细讲解了Spring AOP的核心概念,包括切面、切点、通知等,并提供了在Spring Boot中实现AOP的具体步骤和代码示例。此外,还列举了AOP在日志记录、性能监控、事务管理和安全控制等场景中的实际应用。通过本文,开发者可以快速掌握AOP编程思想及其实践技巧。
|
1月前
|
人工智能 监控 安全
如何快速上手【Spring AOP】?核心应用实战(上篇)
哈喽大家好吖~欢迎来到Spring AOP系列教程的上篇 - 应用篇。在本篇,我们将专注于Spring AOP的实际应用,通过具体的代码示例和场景分析,帮助大家掌握AOP的使用方法和技巧。而在后续的下篇中,我们将深入探讨Spring AOP的实现原理和底层机制。 AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的核心特性之一,它能够帮助我们解决横切关注点(如日志记录、性能统计、安全控制、事务管理等)的问题,提高代码的模块化程度和复用性。
|
1月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
|
8月前
|
XML Java 开发者
Spring Boot中的AOP实现
Spring AOP(面向切面编程)允许开发者在不修改原有业务逻辑的情况下增强功能,基于代理模式拦截和增强方法调用。Spring Boot通过集成Spring AOP和AspectJ简化了AOP的使用,只需添加依赖并定义切面类。关键概念包括切面、通知和切点。切面类使用`@Aspect`和`@Component`注解标注,通知定义切面行为,切点定义应用位置。Spring Boot自动检测并创建代理对象,支持JDK动态代理和CGLIB代理。通过源码分析可深入了解其实现细节,优化应用功能。
415 6
|
7月前
|
XML Java 测试技术
Spring AOP—通知类型 和 切入点表达式 万字详解(通俗易懂)
Spring 第五节 AOP——切入点表达式 万字详解!
372 25
|
7月前
|
XML 安全 Java
Spring AOP—深入动态代理 万字详解(通俗易懂)
Spring 第四节 AOP——动态代理 万字详解!
275 24
|
6月前
|
Java API 微服务
微服务——SpringBoot使用归纳——Spring Boot中的切面AOP处理——Spring Boot 中的 AOP 处理
本文详细讲解了Spring Boot中的AOP(面向切面编程)处理方法。首先介绍如何引入AOP依赖,通过添加`spring-boot-starter-aop`实现。接着阐述了如何定义和实现AOP切面,包括常用注解如`@Aspect`、`@Pointcut`、`@Before`、`@After`、`@AfterReturning`和`@AfterThrowing`的使用场景与示例代码。通过这些注解,可以分别在方法执行前、后、返回时或抛出异常时插入自定义逻辑,从而实现功能增强或日志记录等操作。最后总结了AOP在实际项目中的重要作用,并提供了课程源码下载链接供进一步学习。
752 0
|
6月前
|
Java 开发者 微服务
微服务——SpringBoot使用归纳——Spring Boot中的切面AOP处理——什么是AOP
本文介绍了Spring Boot中的切面AOP处理。AOP(Aspect Oriented Programming)即面向切面编程,其核心思想是分离关注点。通过AOP,程序可以将与业务逻辑无关的代码(如日志记录、事务管理等)从主要逻辑中抽离,交由专门的“仆人”处理,从而让开发者专注于核心任务。这种机制实现了模块间的灵活组合,使程序结构更加可配置、可扩展。文中以生活化比喻生动阐释了AOP的工作原理及其优势。
355 0