0


【JavaEE】面向切面编程AOP是什么-Spring AOP框架的基本使用

【JavaEE】 AOP(1)

在这里插入图片描述

文章目录

【JavaEE】AOP(1)

1. Spring AOP 是什么

1.1 AOP 与 Spring AOP

AOPA spect O riented Programming),是一种思想,即面向切面编程

Spring AOP 则是一个框架,Spring项目中需要引入依赖而使用

  • AOP和Spring AOP的关系就相当于IoC和DI
  • Spring AOP让开发者能够半自动的开发AOP思想下实现的功能

1.2 没有AOP的世界是怎样的

我们要实现用户登录校验的功能,没有实现AOP之前,我们只能这样做:

博客系统登录功能实现,博客传送门:【JavaEE】前后端综合项目-博客系统(下)_s:103的博客-CSDN博客

  1. 前端进入每一个页面的时候,专门发送“登录校验”的请求给后端进行登录校验
  2. 前端发送每一个请求的时候,后端都会自动进行登录校验

无论是哪种,都有这样的特性,就是代码耦合度高,网页的每个功能都要各自实现“登录校验”的代码,这样的坏处是:

  1. 不符合专一设计原则,开发者开发一个功能就应该全心全意,有针对性,这样开发效率也会高点儿~
  2. 耦合度很高,一改则需万改

图示:

在这里插入图片描述

1.3 AOP是什么

刚才说了,AOP是面向切面编程,这是英语直译的结果,不用太多理会~

  • 可以理解为,它是 对某一类事情的集中处理

有了AOP后,我们只需要在代码的某一处配置一下,所有的功能就能实现用户登录校验了,不再需要重复写那些代码了~

  • 也对“登录校验”这个步骤,集中处理了,这个代码开发过程就是面向切面编程

切面可以理解为,切下来的一个方面,一个步骤

切下来后有种藕断丝连的感觉:

藕断丝连 的图像结果

在这里插入图片描述

例如这个图,再等一下讲解AOP的组成之后,回头看可能会有更好的理解~

  • AOP后,功能表现还是左上,但是代码长成右下
  • “登录校验”被切了下来一样

如果把切面看成动作执行者,✂

  • 我们可以将切面形象地比喻为一把剪刀。
  • 剪刀可以在不改变原始材料的情况下,将其切割成不同的形状
  • 同样,切面在编程中也可以在不修改原始代码的情况下,将横切关注点应用到不同的对象或方法上- 在代码实现的时候,将需要的统一处理的功能,“剪”下来进行统一处理
  • 就像剪刀可以将材料切割成不同的形状一样,切面可以将横切关注点切割并应用到代码的不同部分,从而实现代码的模块化和重用

例如,调用功能1的时候,会通过切下来的“面”的匹配对应的“块”,之后进行“组装”~

  • 这样就在调用功能1之前进行了登录校验

好处:

  1. 满足单一设计原则,登录校验的存在,不会改变各个功能的原代码
  2. 耦合度低,改一处等于改一万处

AOP就很适合解决这种功能统一,且是使用的地方较多的功能了

  • AOP的统一处理!

除了登录校验外,还可以实现:

  • 统一1. 日志记录2. 方法执行时间统计3. 返回格式设置4. 异常处理5. 事务开启与提交6. 等等…

也可以说,AOP面向了多个对象

也就是说,AOP是OOP(Object Oriented Programming,面向对象编程)的一个补充和完善

2. Spring AOP 框架的学习

Spring AOP框架内部实现了AOP,开发者按照规则写下的代码,就对应着AOP逻辑

Spring AOP的学习分为以下三步:

  1. AOP的组成
  2. Spring AOP的基本使用
  3. Spring AOP的实现原理

2.1 AOP的组成

2.1.1 Aspect 切面

一个Aspect代表一个“统一处理”,因为统一设计原则,一个切面就是一个功能,例如“登录校验”

  • 包含切点、通知,即有横切逻辑的定义,也有连接点的定义

如果把这些定义,看成“切面”的独一无二的横切面,看成标识,切面就对应那个“块”,也就是功能,所以切面就看成一个功能~

即,AOP是干啥的

2.1.2 Pointcut 切点

规定 这个切面是从哪些,前后端交互接口,“剪”下来的

  • 例如:定义用户登录拦截规则,哪些接口才需要判断用户登录权限

2.1.3 Advice 通知

AOP执行的具体方法

  • 例如:获取用户信息,如果后端有记录对应的登录信息,就说明已经登录,否则没有登录

2.1.4 Join Point 连接点

有可能触发切点的所有点

  • 满足切点规则的所有前后端交互接口

2.2 Spring AOP的基本使用

通过代码实现,你可能会有更切合的体验!

  • 我只讲一种方式,其他感兴趣的可以去学,有需要的去学

2.2.1 引入依赖

你会发现,在IDEA的勾选选入依赖里,没有AOP这个选项,这也很正常,因为并不是所有依赖都包含在其中!

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

在这里插入图片描述

  • 第一次加载会有点慢,注意网络通畅~

2.2.2 定义一个Aspect

在这里插入图片描述

一个切面其实就是一个类,加上@Aspect注解,这个类就是为了实现某个“统一处理”的

  • 需要加五大类注解,这个算是个组键吧,所以用@Component

2.2.3 定义一个Pointcut

在这里插入图片描述

一个方法加上注解@Pointcut,在括号内写一些语句,就定义了一个切点

  • 这个方法可以不返回一切,方法名自定义,没有参数列表,没有代码体…,如果加了会怎么样感兴趣可以去了解,但是暂时用不到~
  • 括号内部的语句是有对应的语法的,这里了解这句话的含义,举一反三就好,语法是啥感兴趣可以去了解
execution(* com.example.demo.controller.TestController.*(..))

,解析:

  1. execution(),执行括号中提及的方法,就触发切点
  2. *,不限制方法权限(public、private…),全部方法
  3. com.example.demo.controller.TestController.,针对的类的位置
  4. *(..),代表类中的所有方法 1. *,所有的方法名2. (..),任意的参数列表

2.2.4 Join Point

对于连接点,则是刚才切点对应的前后端交互接口:

  1. /test/say_hi
  2. /test/say_hello

我们在这里不需要在意其他业务,只需要专注开发,而Aspect的代码开发里则面向了这几个连接点!

在这里插入图片描述

为了与通知的方法区分,我在控制台打印信息:

在这里插入图片描述

2.2.5 定义Advice

通知就是访问对应接口的时候,要执行的业务,分为五种通知:

  1. 前置通知
  2. 后置通知
  3. 环绕通知
  4. 返回通知
  5. 异常通知

顾名思义,区分就是,通知执行的时机~

对应五种注解(加在一个方法的上面,此方法被称为通知方法):

  1. @Before,通知方法在目标方法执行之前执行
  2. @After,通知方法在目标方法执行之后执行
  3. @Around,通知包裹了目标方法,自定义行为
  4. @AfterReturning,通知方法在目标方法返回后调用
  5. @AfterThrowing,通知方法在目标抛出异常后调用
2.2.5.1 前置通知,后置通知与返回通知
// 前置通知@Before("pointcut()")publicvoiddoBefore(){
    System.out.println("执⾏ Before ⽅法");}// 后置通知@After("pointcut()")publicvoiddoAfter(){
    System.out.println("执⾏ After ⽅法");}// return 之前通知@AfterReturning("pointcut()")publicvoiddoAfterReturning(){
    System.out.println("执⾏ AfterReturning ⽅法");}

注解括号内代表,对应的切点方法

效果:

在这里插入图片描述

按多次:

在这里插入图片描述

2.2.5.2 异常通知
// 抛出异常之前通知@AfterThrowing("pointcut()")publicvoiddoAfterThrowing(){
    System.out.println("执⾏ doAfterThrowing ⽅法");}

对于异常通知的触发案例,暂时不介绍

2.2.5.3 环绕通知

自定义行为的通知,可以模拟其他通知~

@Around("pointcut()")public Object doAround(ProceedingJoinPoint joinPoint)throws Throwable {
    Object object = null;
    System.out.println("Around方法开始执行");
    object = joinPoint.proceed();
    System.out.println("Around方法结束执行");return object;}

ProceedingJoinPoint这个类的对象joinPoint,代表着连接点对象

  • 调用proceed()方法代表进行这个连接点对于的方法 - try catch后能模拟异常通知

记得返回这个object,不然框架处理不了返回值作为响应!

  • 默认执行后的返回值是交给框架的,而这里是交给object~

效果:

在这里插入图片描述

这样也看出了优先级~

环绕通知,更灵活

  • 更好的实现“原子性”,通过加锁就行,如执行时间统计…
  • 如后置通知方法需要跟前置通知方法有“数据交互”

2.3 Spring AOP的实现原理

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

2.3.1 动态代理

原本访问目标对象:

在这里插入图片描述

AOP之后:

在这里插入图片描述

就像fiddler一样,代理了请求,期间保留了其信息…

通过代理类,目标对象Join Point何时执行目标方法,代理说了算

  • 在环绕通知的时候体现尤为显著
  • 代理类 根据 Aspect 进行对应的行为

我们不通过Spring AOP框架,我们只能自己实现AOP才能实现在目标方法调用之前和调用之后…时机进行一些业务!

2.3.2 Spring AOP动态代理组成

  1. JDK Proxy- 代理类必须实现某个接口,才能使用JDK Proxy- 底层有用到反射…
  2. CGLIB- 通过实现代理类的子类来实现动态代理,只能代理能被继承的类

通过这两种动态代理的实现~

  • 单独一种有局限性

2.3.3 JDK Proxy 与 CGLIB的区别

  1. 出身不同- JDK Proxy 来自java- CGLIB 来自第三方
  2. 实现不同- JDK Proxy,通过代理类必须实现某个接口- CGLIB,通过实现代理类的子类
  3. 性能不同- jdk 7/7+,JDK Proxy性能略高于CBLIB- jdk7-,CBLIB性能远高于JDK Proxy

文章到此结束!谢谢观看
可以叫我 小马,我可能写的不好或者有错误,但是一起加油鸭🦆

不得不提的是,切面、切点、连接点、通知,这些名词,不要纠结他们为啥这么叫,咱们知道他们的含义,知道Spring AOP怎么开发就行了~

代码:spring_aop-demo · 游离态/马拉圈2023年8月 - 码云 - 开源中国 (gitee.com)


标签: java-ee spring java

本文转载自: https://blog.csdn.net/Carefree_State/article/details/132400492
版权归原作者 s:103 所有, 如有侵权,请联系我们删除。

“【JavaEE】面向切面编程AOP是什么-Spring AOP框架的基本使用”的评论:

还没有评论