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

【JavaEE】 AOP(1)

文章目录

  • 【JavaEE】AOP(1)
    • [1. Spring AOP 是什么](#1. Spring AOP 是什么)
      • [1.1 AOP 与 Spring AOP](#1.1 AOP 与 Spring AOP)
      • [1.2 没有AOP的世界是怎样的](#1.2 没有AOP的世界是怎样的)
      • [1.3 AOP是什么](#1.3 AOP是什么)
    • [2. Spring AOP 框架的学习](#2. Spring AOP 框架的学习)
      • [2.1 AOP的组成](#2.1 AOP的组成)
        • [2.1.1 Aspect 切面](#2.1.1 Aspect 切面)
        • [2.1.2 Pointcut 切点](#2.1.2 Pointcut 切点)
        • [2.1.3 Advice 通知](#2.1.3 Advice 通知)
        • [2.1.4 Join Point 连接点](#2.1.4 Join Point 连接点)
      • [2.2 Spring AOP的基本使用](#2.2 Spring AOP的基本使用)
        • [2.2.1 引入依赖](#2.2.1 引入依赖)
        • [2.2.2 定义一个Aspect](#2.2.2 定义一个Aspect)
        • [2.2.3 定义一个Pointcut](#2.2.3 定义一个Pointcut)
        • [2.2.4 Join Point](#2.2.4 Join Point)
        • [2.2.5 定义Advice](#2.2.5 定义Advice)
          • [2.2.5.1 前置通知,后置通知与返回通知](#2.2.5.1 前置通知,后置通知与返回通知)
          • [2.2.5.2 异常通知](#2.2.5.2 异常通知)
          • [2.2.5.3 环绕通知](#2.2.5.3 环绕通知)
      • [2.3 Spring AOP的实现原理](#2.3 Spring AOP的实现原理)
        • [2.3.1 动态代理](#2.3.1 动态代理)
        • [2.3.2 Spring AOP动态代理组成](#2.3.2 Spring AOP动态代理组成)
        • [2.3.3 JDK Proxy 与 CGLIB的区别](#2.3.3 JDK Proxy 与 CGLIB的区别)

【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这个选项,这也很正常,因为并不是所有依赖都包含在其中!

xml 复制代码
<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 前置通知,后置通知与返回通知
java 复制代码
// 前置通知
@Before("pointcut()")
public void doBefore(){
    System.out.println("执⾏ Before ⽅法");
}
// 后置通知
@After("pointcut()")
public void doAfter(){
    System.out.println("执⾏ After ⽅法");
}
// return 之前通知
@AfterReturning("pointcut()")
public void doAfterReturning(){
    System.out.println("执⾏ AfterReturning ⽅法");
}

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

效果:

按多次:

2.2.5.2 异常通知
java 复制代码
// 抛出异常之前通知
@AfterThrowing("pointcut()")
public void doAfterThrowing(){
    System.out.println("执⾏ doAfterThrowing ⽅法");
}

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

2.2.5.3 环绕通知

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

java 复制代码
@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)


相关推荐
Leslie_Lei3 分钟前
Hutool-Java工具库
java·hutool
瓜牛_gn7 分钟前
spring框架基础
spring
斗-匕18 分钟前
Spring & Spring Boot 常用注解总结
java·spring boot·spring
夏微凉.20 分钟前
【JavaEE进阶】Spring 事务和事务传播机制
java·数据库·sql·mysql·spring·java-ee
吴冰_hogan24 分钟前
nacos集群源码解析-cp架构
java·spring boot·spring·架构·服务发现·springcloud
阿七想学习25 分钟前
数据结构《链表》
java·开发语言·数据结构·学习·链表
Yaml425 分钟前
Java的六大排序
java·算法·排序算法
XiaoLiuLB28 分钟前
Tomcat NIO 配置实操指南
java·tomcat·nio
Be_Somebody31 分钟前
[这可能是最好的Spring教程!]Maven的模块管理——如何拆分大项目并且用parent继承保证代码的简介性
java·spring boot·spring·spring入门
一个数据小开发1 小时前
业务开发问题之ConcurrentHashMap
java·开发语言·高并发·map