Spring AOP

简介: Spring AOP

一、简介Spring AOP

APO指的是面向切面编程,与OOP(面向对象)类似,是对某一类事情的集中处理。那么Spring AOP就是这个思想的具体实现。

例如对用户的登录权限进行校验,在没有AOP之前就需要在判断用户是否是登录状态的页面都需要实现或者调用验证用户登录的方法,但是在使用AOP之后,就只需要在某一处进行配置,所有判断用户是否是登录状态的页面就全都可以实现用户登录验证了。

二、Spring AOP的相关概念

切面:定义AOP处理的统一功能,这个功能就叫做切面,比如验证用户登录的功能就可以成为一个切面,切面由切点和通知组成。

连接点:触发AOP(拦截方法)的点就可以称之为连接点。

切点:定义AOP的拦截规则。

通知:规定AOP的执行时机和执行方法,通知可以有前置通知、后置通知、抛出异常后通知、返回数据后通知以及环绕通知

通俗点来讲:切面就是设想一个大型活动,那么切点就是活动策划书,通知就是活动的具体执行,连接点就是触发活动策划中的一些事件。

三、Spring AOP的具体实现

实现Spring AOP具体可以分为如下几步:

  1. 在项目中添加Spring AOP框架;
  2. 定义切面;
  3. 定义切点;
  4. 实现通知。

添加Spring AOP框架

Spring Boot框架中并没有Spring AOP框架 ,需要去maven中心库进行引入与自己项目版本相匹配的依赖。

<!-- https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-aop -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
    <version>3.0.5</version>
</dependency>

定义切面

定义切面时就需要创建一个类,添加@Aspect注解表示该类是一个切面。

例如定义一个UserAspect类:

@Aspect //表示当前类是一个切面
@Component
public class UserAspect {
    
}

定义切点

切点是切面中的一个方法,需要使用@PointCut注解,里面需要传入一个AspectJ表达式,AspectJ是一个第三方库,但是由于比较好用,SpringAOP也是兼容的。

如下为AspectJ表达式:

在AspectJ表达式中也可以使用如下的通配符:

  • *:可以匹配任意的内容,可以使用在返回值、包名、类名以及方法名。
  • ..:匹配任意字符,常用在方法的参数中,若在类上使用就需要配合*来使用。
  • +:用于匹配指定类及其所有的子类。

例如,在UserAspect切面中定义如下的切点:

    @Pointcut("execution(* com.example.spring_aop.controller.UserController.*(..))")
    public void pointCut(){}

实现通知

在controller包下定义UserController类,并定义sayHi方法:

@RestController
@RequestMapping("/user")
public class UserController {
    @RequestMapping("/sayhi")
    public String  sayHi(){
        return "hi!";
    }
}

前置通知

表示在执行目标方法之前执行该方法,使用@Before注解,其中需要传入的参数是切点的方法名。

@Before("pointCut()")
    public void doBefore(){
        System.out.println("前置方法已经执行");
    }

那么访问UserController的sayhi页面:

并且前置方法也被执行了:

后置通知

在执行目标方法之后执行该方法,使用@After注解,同样需要传入切点的方法名。

@After("pointCut()")
public void doAfter(){
    System.out.println("后置方法已经执行");
}

那么再次访问sayhi页面就会有如下:

 

返回数据后通知

在目标方法返回数据后执行该方法,如果没有返回数据该方法就不会执行,该方法的执行通常是在后置通知之前执行,使用@AfterReturning注解,传入的参数同样也是切点的方法名。

那么再次访问sayhi页面之后就出现如下所示:

抛出异常后通知

在目标方法抛出异常后执行该方法,如果没有抛出异常该方法就不会执行,使用@AfterThrowing注解,传入的参数同样也是切点的方法名。

例如将sayhi方法修改如下:

@RequestMapping("/sayhi")
    public String  sayHi() throws Exception{
        int num = 1/0;
        return "hi!";
    }

那么再次对sayhi进行访问时就会出现如下:

由于在返回数据之前就已经抛出异常了就没有正常返回数据,因此就不会执行返回数据后的通知。

环绕通知

包裹了被通知的方法,在被通知方法通知之前和调用之后使用自定义方法,使用@Around注解,传入的参数同样也是切点的方法名。


该通知相比之前的通知较为复杂,返回值是Object,传入的参数是ProceedingJoinPoint对象,其中方法体中调用该对象的proceed()方法。

@Around("pointCut()")
public Object doAround(ProceedingJoinPoint joinPoint){
        Object result = null;
        System.out.println("环绕通知开始");
        try {
            result = joinPoint.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("环绕通知结束");
        return result;
    }

对sayhi进行访问时就会出现如下所示:

那么利用环绕通知可以求目标方法的执行时间:

@Around("pointCut()")
    public Object doAround(ProceedingJoinPoint joinPoint){
        Object result = null;
        System.out.println("环绕通知开始");
        try {
            long start = System.currentTimeMillis();
            result = joinPoint.proceed();
            long end = System.currentTimeMillis();
            System.out.println(joinPoint.getSignature().getName()+"方法的执行时间:"+(end-start)+"ms");
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("环绕通知结束");
        return result;
    }

运行结果:

但是这样写只适合单线程的情况,可以在环绕方法中定义一个StopWath对象, 调用其start方法以及stop方法来计算目标方法的运行时间。

@Around("pointCut()")
    public Object doAround(ProceedingJoinPoint joinPoint){
        Object result = null;
        System.out.println("环绕通知开始");
        StopWatch stopWatch = new StopWatch();
        try {
            stopWatch.start();
            result = joinPoint.proceed();
            stopWatch.stop();
            System.out.println(joinPoint.getSignature().getName()+"方法的执行时间:"+stopWatch.getTotalTimeMillis()+"ms");
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("环绕通知结束");
        return result;
    }

四、Spring AOP的实现原理

不引入AOP,就是前端和后端直接进行交互,但是在引入AOP之后,AOP就是中间的代理商,进行动态代理,Spring对AOP仅支持方法级别的拦截,AOP在运行期开始代理

Spring AOP ⽀持 JDK Proxy 和 CGLIB ⽅式实现动态代理:

JDK Proxy(JDK动态代理);

GGLIB Proxy:默认情况下使用GGLIB Proxy进行动态代理,GGLIB Proxy通过继承代理对象来实现动态代理,所以GGLIB Proxy不能代理被final修饰的类,就需要使用JDK Proxy进行代理。

目录
相关文章
|
1天前
|
存储 安全 Java
Spring Boot 3 集成Spring AOP实现系统日志记录
本文介绍了如何在Spring Boot 3中集成Spring AOP实现系统日志记录功能。通过定义`SysLog`注解和配置相应的AOP切面,可以在方法执行前后自动记录日志信息,包括操作的开始时间、结束时间、请求参数、返回结果、异常信息等,并将这些信息保存到数据库中。此外,还使用了`ThreadLocal`变量来存储每个线程独立的日志数据,确保线程安全。文中还展示了项目实战中的部分代码片段,以及基于Spring Boot 3 + Vue 3构建的快速开发框架的简介与内置功能列表。此框架结合了当前主流技术栈,提供了用户管理、权限控制、接口文档自动生成等多项实用特性。
23 8
|
2月前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
2月前
|
监控 安全 Java
什么是AOP?如何与Spring Boot一起使用?
什么是AOP?如何与Spring Boot一起使用?
76 5
|
2月前
|
Java 开发者 Spring
深入解析:Spring AOP的底层实现机制
在现代软件开发中,Spring框架的AOP(面向切面编程)功能因其能够有效分离横切关注点(如日志记录、事务管理等)而备受青睐。本文将深入探讨Spring AOP的底层原理,揭示其如何通过动态代理技术实现方法的增强。
80 8
|
2月前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。
|
2月前
|
XML 监控 安全
深入调查研究Spring AOP
【11月更文挑战第15天】
50 5
|
2月前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
51 4
|
3月前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
57 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
2月前
|
Java Spring
[Spring]aop的配置与使用
本文介绍了AOP(面向切面编程)的基本概念和核心思想。AOP是Spring框架的核心功能之一,通过动态代理在不修改原代码的情况下注入新功能。文章详细解释了连接点、切入点、通知、切面等关键概念,并列举了前置通知、后置通知、最终通知、异常通知和环绕通知五种通知类型。
46 1
|
4月前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP