Spring AOP讲解

目录

Spring AOP概念

AOP适用场景

AOP 组成

1、切面(Aspect)

2、切点(Pointcut)

3、通知(Advice)

4、连接点(Join Point)

Spring AOP实现

添加AOP框架依赖

定义切面和切点

AspectJ表达式说明

 定义通知

SpringAOP的原理

织入(代理的生成时机)

动态代理

JDK 及 CGLIB 的方式的异同点


Spring AOP概念

AOP :面向切面编程,它是⼀种思想,它是对某⼀类事情的集中处理。

        就比如说在进行登录注册界面验证时,凡是需要使用到注册登录的方法,都需要添加登录注册验证代码。

        因此在使用AOP之后,我们只需要在某⼀处配置⼀下,所有需要判断⽤户登录⻚⾯(中的方法)就全部可以实现⽤户登录验证了,不再需要每个⽅法中都写相同的⽤户登录验证了。

Spring AOP :AOP 是一种思想,而Spring AOP是一个框架,实现了对AOP思想的实现。

AOP适用场景

        对于一些功能统一而且许多地方需要经常使用,这时候就可以使用AOP来进行统一进行处理了。

AOP的实现功能:

        1.统一的用户登录注册判断

        2.统一日志记录

        3.统一方法执行时间统计

        4.统一返回格式设置

        5.统一的异常处理

        6.事务的开启和提交等

AOP相当于是一个拦截器,需要经过AOP的验证才可访问服务器,效果图如下:

        

AOP 组成

1、切面(Aspect)

切面是包含了:通知、切点和切面的类,相当于 AOP 实现的某个功能的集合
通俗的说, 切面在程序里面就是处理一个某方面具体问题的类,类中的方法就是切点和通知

2、切点(Pointcut

用来进行主动拦截的规则(配置)

切点相当于保存了众多连接点的⼀个集合(如果把切点看成⼀个表,⽽连接点就是表中⼀条⼀条 的数据)

3、通知(Advice)

切面所需要完成的工作就是通知。

通知:定义了切⾯是什么,何时使⽤,其描述了切面要完成的⼯作,还解决何时执行这个工作的问题。
Spring 切面类中,可以在方法上使用注解,会设置方法为通知方法,在满足条件后会通知本方法进⾏调⽤。

常见的通知注解如下:

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

4、连接点(Join Point

        应⽤执⾏过程中能够插⼊切⾯的⼀个点,这个点可以是⽅法调⽤时,抛出异常时,甚⾄修改字段
时。切⾯代码可以利⽤这些点插⼊到应⽤的正常流程之中,并添加新的行为
        通俗的理解为:可能会触发AOP规则的所有请求

Spring AOP实现

添加AOP框架依赖

进入中央仓库官网:​​​​​​​https://mvnrepository.com/  并在搜索框中搜索AOP 

在项目中pom.xml查看自己的Spring Boot项目版本

 

 

2.7.12版本的AOP依赖

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

 

定义切面和切点

定义切面

@Aspect  //告诉框架是一个切面
@Component //随着框架启动而启动
public class UserAspect {

}

定义切点

 /*
    * 切点(配置拦截规则)
    * */

    @Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
    public void pointcut(){

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

 

AspectJ表达式说明

AspectJ ⽀持三种通配符
* :匹配任意字符,只匹配⼀个元素(包,类,或⽅法,⽅法参数)
.. :匹配任意字符,可以匹配多个元素 ,在表示类时,必须和 * 联合使⽤。
+ :表示按照类型匹配指定类的所有类,必须跟在类名后⾯,如 com.cad.Car+ ,表示继承该类的所有子类包括本身

AspectJ 表达式由切点函数组成,其中 execution() 是最常⽤的切点函数,⽤来匹配方法,语法为:
        execution(<修饰符><返回类型><包.类.方法(参数)><异常>)

注意:返回类型与包之间用一个空格隔开

示例: execution(* com.cad.demo.User.*(..)) :匹配 User 类里的所有方法
修饰符和异常可以省略,具体含义如下:

修饰符(一般省略):

public    公共方法

     *          任意

返回类型(一般不省略):

void  返回没有值

String  返回值字符串

*              任意

包:(一般情况下要有但是可以省略)

  1. com.gyf.crm      固定包
  2. com.gyf.crm.*.service      crm包下面子包任意 (例如: com.gyf.crm.staff.service)
  3. com.gyf.crm..        crm包下面的所有子包 (含自己)
  4. com.gyf.crm.*.service..         crm包下面任意子包,固定目录service,service目录任意包

类(一般情况下要有但是可以省略):

UserServicelmpl     指定类,

*Impl        以Impl结尾,

User        以User开头

   *                 任意

方法名(不可省略):

addUser        固定方法

add*        以add开头

*Do        以Do结尾

  *                任意

参数:

() 无参

(int) 一个参数

(int,int) 两个参数

(..)   任意参数

异常:throws 一般省略。

 定义通知

 

定义通知代码:

package com.example.demo.AOP;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

@Aspect  //告诉框架是一个切面
@Component //随着框架启动而启动
public class UserAspect {

    /*
    * 切点(配置拦截规则)
    * */

    @Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
    public void pointcut(){

    }

    /*
    * 前置通知*/
    @Before("pointcut()")
    public void beforeAdvice(){
        System.out.println("执行了前置通知~");
    }

    /*
    * 后置通知*/
    @After("pointcut()")
    public void afterAdvice(){
        System.out.println("执行了后置通知~");
    }

    /*
    * 环绕通知*/
    @Around("pointcut()")
    public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("进入了环绕通知~");
        Object obj = null;
        obj = joinPoint.proceed();
        System.out.println("退出环绕通知~");
        return obj;
    }

}

拦截代码:

package com.example.demo.controller;

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/hello")
public class UserController {

    @RequestMapping("/hi")
    public String sayHi(String name){
        System.out.println("执行了sayHi方法");
        return "hi," + name;
    }
}

运行启动类

 

 

SpringAOP的原理

          Spring AOP 是构建在动态代理基础上,因此 Spring 对 AOP 支持局限于方法级别的拦截,Spring AOP 支持 JDK Proxy 和 CGLIB 方式实现动态代理

        我们在前后端交互的时候没有代理的时候前后端是直接进行交互的,但是这样我们就需要去校验前端的一些数据等,如果我们有代理的话那么前端会先将数据传到代理代理做一个处理然后代理再将数据给后端,如此一来我们就可以专注于代码逻辑了。

织入(代理的生成时机)

        织⼊是把切⾯应⽤到⽬标对象并创建新的代理对象的过程,切⾯在指定的连接点被织⼊到⽬标对象中。

在⽬标对象的⽣命周期⾥有多个点可以进⾏织⼊:

  • 编译期:切⾯在⽬标类编译时被织⼊。这种⽅式需要特殊的编译器。AspectJ的织⼊编译器就是以这种⽅式织⼊切⾯的。
  • 类加载期:切⾯在⽬标类加载到JVM时被织⼊。这种⽅式需要特殊的类加载(ClassLoader),它可以在⽬标类被引⼊应⽤之前增强该⽬标类的字节码。AspectJ5的加载时织⼊(load-time weaving. LTW)就⽀持以这种⽅式织⼊切⾯。
  • 运⾏期:切⾯在应⽤运⾏的某⼀时刻被织⼊。⼀般情况下,在织⼊切⾯时,AOP容器会为⽬标对象动态创建⼀个代理对象。SpringAOP就是以这种⽅式织⼊切⾯的。

动态代理

        动态代理指的是在程序运行时动态地创建一个实现特定接口或一组接口的对象,该对象可以拦截并处理所有传递给它的方法调用。动态代理通常是通过在运行时生成字节码来实现的,从而避免了在编译时手工编写代理类的繁琐过程。

        我们学习 Spring 框架中的AOP,主要基于两种⽅式:JDK 及 CGLIB 的⽅式。这两种⽅式的代理⽬标都是被代理类中的⽅法,在运⾏期,动态的织⼊字节码⽣成代理类。

JDK 及 CGLIB 的方式的异同点

相同点:

  1. 它们都使用了反射机制来实现动态代理。
  2. 都可以在运行时动态生成代理对象,在代理对象中调用委托类的方法。

区别:

  1. JDK 是官方提供的,CGLIB 是第三方提供的。
  2. CGLIB 比 JDK 更高效。
  3. JDK 实现,要求被代理类必须实现接⼝,之后是通过 InvocationHandler 及 Proxy,在运⾏时动态的在内存中⽣成了代理类对象,该代理对象是通过实现同样的接⼝实现(类似静态代理接⼝实现的⽅式),只是该代理类是在运⾏期时,动态的织⼊统⼀的业务逻辑字节码来完成。CGLIB 实现,被代理类可以不实现接⼝,是通过继承被代理类,在运⾏时动态的⽣成代理类对象所以相对而言更加的灵活。
  4. CGLIB 是通过实现继承代理对象来实现动态代理的。如果代理的对象是最终类(被 final 修饰的类),Spring AOP 才会去调用 JDK 的方式生成 动态代理。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值