【Spring篇】AOP

news2024/11/15 21:05:50

🍓系列专栏:Spring系列专栏

🍉个人主页:个人主页

目录

一、AOP简介

1.什么是AOP?

2.AOP作用

3.AOP核心概念

二、AOP入门案例

1.需求分析

2.思路分析 

 3.环境准备

4.AOP实现步骤

三、AOP工作流程 

1.AOP工作流程

2.AOP核心概念

四、AOP配置管理

1.AOP切入点表达式

1,语法格式

2.通配符

3.书写技巧

 2.AOP通知类型

1.类型介绍

2.环境准备

3.通知类型的使用


一、AOP简介

前面我们在介绍 Spring 的时候说过, Spring 有两个核心的概念,一个是 IOC/DI ,一个是 AOP
前面已经对 IOC/DI 进行了系统的学习,接下来要学习它的另一个核心内容,就是 AOP
对于 AOP, 我们前面提过一句话是 : AOP 是在不改原有代码的前提下对其进行增强。
对于下面的内容,我们主要就是围绕着这一句话进行展开学习,主要学习两方面内容 AOP 核心概
, AOP 作用

1.什么是AOP?

  • AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程 序结构。
  • OOP(Object Oriented Programming)面向对象编程
我们都知道 OOP 是一种编程思想,那么 AOP 也是一种编程思想,编程思想主要的内容就是指导程序员该如何编写程序,所以它们两个是不同的编程范式

2.AOP作用

作用 : 在不惊动原始设计的基础上为其进行功能增强,前面咱们有技术就可以实现这样的功能即代
理模式。

3.AOP核心概念

为了能更好的理解 AOP 的相关概念,我们准备了一个环境,整个环境的内容我们暂时可以不用关注,最主要的类为: BookDaoImpl
@Repository
public class BookDaoImpl implements BookDao {
public void save() {
//记录程序当前执行执行(开始时间)
Long startTime = System.currentTimeMillis();
 //业务执行万次
 for (int i = 0;i<10000;i++) {
 System.out.println("book dao save ...");
 }
 //记录程序当前执行时间(结束时间)
 Long endTime = System.currentTimeMillis();
 //计算时间差
 Long totalTime = endTime-startTime;
 //输出信息
 System.out.println("执行万次消耗时间:" + totalTime + "ms");
 }
 public void update(){
 System.out.println("book dao update ...");
 }
 public void delete(){
 System.out.println("book dao delete ...");
 }
 public void select(){
 System.out.println("book dao select ...");
 }
 }
代码的内容相信大家都能够读懂,对于 save 方法中有计算万次执行消耗的时间。
当在 App 类中从容器中获取 bookDao 对象后,分别执行其 save , delete , update select 方法后会
有如下的打印结果 :

这个时候,我们就应该有些疑问 ?
  • 对于计算万次执行消耗的时间只有save方法有,为什么deleteupdate方法也会有呢?
  • deleteupdate方法有,那什么select方法为什么又没有呢?
这个案例中其实就使用了 Spring AOP ,在不惊动 ( 改动 ) 原有设计 ( 代码 ) 的前提下,想给谁添加功能就给谁添加。这个也就是Spring 的理念:
  • 无入侵式/无侵入式
说了这么多, Spring 到底是如何实现的呢 ?
(1) 前面一直在强调, Spring AOP 是对一个类的方法在不进行任何修改的前提下实现增强。对于上面的案例中BookServiceImpl 中有 save , update , delete select 方法 , 这些方法我们给起了一
个名字叫 连接点
(2) BookServiceImpl 的四个方法中, update delete 只有打印没有计算万次执行消耗时间,
但是在运行的时候已经有该功能,那也就是说 update delete 方法都已经被增强,所以对于需要增
强的方法我们给起了一个名字叫 切入点
(3) 执行 BookServiceImpl update delete 方法的时候都被添加了一个计算万次执行消耗时间的功能,将这个功能抽取到一个方法中,换句话说就是存放共性功能的方法,我们给起了个名字叫
(4) 通知是要增强的内容,会有多个,切入点是需要被增强的方法,也会有多个,那哪个切入点需要添加哪个通知,就需要提前将它们之间的关系描述清楚,那么对于通知和切入点之间的关系描述,我们给起了个名字叫 切面
(5) 通知是一个方法,方法不能独立存在需要被写在一个类中,这个类我们也给起了个名字叫 通知类 至此AOP 中的核心概念就已经介绍完了,总结下 :

连接点(JoinPoint)程序执行过程中的任意位置,粒度为执行方法、抛出异常、设置变量等

  • SpringAOP中,理解为方法的执行

切入点(Pointcut):匹配连接点的式子

  • 在SpringAOP中,一个切入点可以描述一个具体方法,也可也匹配多个方法
  • 一个具体的方法:com.itheima.dao包下的BookDao接口中的无形参无返回值的save方法
  • 匹配多个方法:所有的save方法,所有的get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法
  • 连接点范围要比切入点范围大,是切入点的方法也一定是连接点,但是是连接点的方法就不一 定要被增强,所以可能不是切入点。

通知(Advice):在切入点处执行的操作,也就是共性功能

  • 在SpringAOP中,功能最终以方法的形式呈现

通知类:定义通知的类

切面(Aspect):描述通知与切入点的对应关系。

二、AOP入门案例

1.需求分析

案例设定:测算接口执行效率,但是这个案例稍微复杂了点,我们对其进行简化。
简化设定:在方法执行前输出当前系统时间。
对于 SpringAOP 的开发有两种方式, XML 注解 ,我们使用哪个呢 ?
因为现在注解使用的比较多,所以本次课程就采用注解完成 AOP 的开发。
总结需求为 : 使用 SpringAOP 的注解方式完成在方法执行的前打印出当前系统时间。

2.思路分析 

需求明确后,具体该如何实现,都有哪些步骤,我们先来分析下 :
1. 导入坐标 (pom.xml)
2. 制作连接点 ( 原始操作, Dao 接口与实现类 )
3. 制作共性功能 ( 通知类与通知 )
4. 定义切入点
5. 绑定切入点与通知关系 ( 切面 )

 3.环境准备

创建一个Maven项目

pom.xml添加Spring依赖

<dependencies>
 <dependency>
 <groupId>org.springframework</groupId>
 <artifactId>spring-context</artifactId> 
<version>5.2.10.RELEASE</version> 
</dependency> 
</dependencies>
添加 BookDao BookDaoImpl
public interface BookDao {
    public void save();
    public void update();
}
@Repository
public class BookDaoImpl implements BookDao {

    public void save() {
        System.out.println(System.currentTimeMillis());
        System.out.println("book dao save ...");
    }

    public void update(){
        System.out.println("book dao update ...");
    }
}
创建 Spring 的配置类
@Configuration
@ComponentScan("com.itheima")

public class SpringConfig {
}
编写 App 运行类
public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        BookDao bookDao = ctx.getBean(BookDao.class);
        bookDao.save();

    }
}
最终创建好的项目结构如下 :

说明 :
  • 目前打印save方法的时候,因为方法中有打印系统时间,所以运行的时候是可以看到系统时间
  • 对于update方法来说,就没有该功能
  • 我们要使用SpringAOP的方式在不改变update方法的前提下让其具有打印系统时间的功能。

4.AOP实现步骤

步骤 1: 添加依赖
pom.xml
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>

因为 spring - context 中已经导入了 spring - aop , 所以不需要再单独导入 spring - aop
导入 AspectJ jar ,AspectJ AOP 思想的一个具体实现, Spring 有自己的 AOP 实现,但是相比于AspectJ 来说比较麻烦,所以我们直接采用 Spring 整合 ApsectJ 的方式进行 AOP 开发。
步骤 2: 定义接口与实现类
环境准备的时候, BookDaoImpl 已经准备好,不需要做任何修改
步骤 3: 定义通知类和通知
通知就是将共性功能抽取出来后形成的方法,共性功能指的就是当前系统时间的打印。
public class MyAdvice {
public void method(){
System.out.println(System.currentTimeMillis());
}
}
类名和方法名没有要求,可以任意。
步骤 4: 定义切入点
BookDaoImpl 中有两个方法,分别是 save update ,我们要增强的是 update 方法,该如何定义呢 ?
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
public void method(){
System.out.println(System.currentTimeMillis());
}
}
说明 :
  • 切入点定义依托一个不具有实际意义的方法进行,即无参数、无返回值、方法体无实际逻辑。
  • execution及后面编写的内容,后面会有章节专门去学习。

步骤 5: 制作切面
切面是用来描述通知和切入点之间的关系,如何进行关系的绑定 ?
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Before("pt()")
public void method(){
System.out.println(System.currentTimeMillis());
}
}
绑定切入点与通知关系,并指定通知添加到原始连接点的具体执行 位置

说明 : @Before 翻译过来是之前,也就是说通知会在切入点方法执行之前执行,除此之前还有其他四种类型,后面会讲。
步骤 6: 将通知类配给容器并标识其为切面类
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Before("pt()")
public void method(){
System.out.println(System.currentTimeMillis());
}
}

步骤 7: 开启注解格式 AOP 功能
@Configuration
@ComponentScan("com.itheima")
@EnableAspectJAutoProxy
public class SpringConfig {
}
步骤 8: 运行程序
public class App {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(SpringConfig.class);
BookDao bookDao = ctx.getBean(BookDao.class);
bookDao.update();
}
}
看到在执行 update 方法之前打印了系统时间戳,说明对原始方法进行了增强, AOP 编程成功。

 

三、AOP工作流程 

AOP 的入门案例已经完成,对于刚才案例的执行过程,我们就得来分析分析,这一节我们主要讲解两个知识点: AOP 工作流程 AOP 核心概念 。其中核心概念是对前面核心概念的补充。

1.AOP工作流程

由于AOP是基于Spring容器管理的bean做的增强,所以整个工作过程需要从Spring加载bean说起:

流程 1:Spring 容器启动
  • 容器启动就需要去加载bean,哪些类需要被加载呢?
  • 需要被增强的类,如:BookServiceImpl
  • 通知类,如:MyAdvice
  • 注意此时bean对象还没有创建成功
流程 2: 读取所有切面配置中的切入点

 上面这个例子中有两个切入点的配置,但是第一个ptx()并没有被使用,所以不会被读取。

流程 3: 初始化 bean
判定 bean 对应的类中的方法是否匹配到任意切入点
  • 注意第1步在容器启动的时候,bean对象还没有被创建成功。
  • 要被实例化bean对象的类中的方法和切入点进行匹配

匹配失败,创建原始对象 , UserDao
  • 匹配失败说明不需要增强,直接调用原始对象的方法即可。
匹配成功,创建原始对象( 目标对象 )的 代理 对象 , : BookDao
  • 匹配成功说明需要对其进行增强
  • 对哪个类做增强,这个类对应的对象就叫做目标对象
  • 因为要对目标对象进行功能增强,而采用的技术是动态代理,所以会为其创建一个代理对象
  • 最终运行的是代理对象的方法,在该方法中会对原始方法进行功能增强

流程 4: 获取 bean 执行方法
获取的 bean 是原始对象时,调用方法并执行,完成操作
获取的 bean 是代理对象时,根据代理对象的运行模式运行原始方法与增强的内容,完成操作
验证容器中是否为代理对象
为了验证 IOC 容器中创建的对象和我们刚才所说的结论是否一致,首先先把结论理出来 :
如果目标对象中的方法会被增强,那么容器中将存入的是目标对象的代理对象
如果目标对象中的方法不被增强,那么容器中将存入的是目标对象本身。
验证思路
1. 要执行的方法,不被定义的切入点包含,即不要增强,打印当前类的 getClass() 方法
2. 要执行的方法,被定义的切入点包含,即要增强,打印出当前类的 getClass() 方法
3. 观察两次打印的结果
步骤 1: 修改 App , 获取类的类型
public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        BookDao bookDao = ctx.getBean(BookDao.class);
//        bookDao.update();
        System.out.println(bookDao);
        System.out.println(bookDao.getClass());
    }
}
步骤 2: 修改 MyAdvice 类,不增强
因为定义的切入点中,被修改成 update1 , 所以 BookDao 中的 update 方法在执行的时候,就不会被增强, 所以容器中的对象应该是目标对象本身。
@Component
//设置当前类为切面类类
@Aspect
public class MyAdvice {
    //设置切入点,要求配置在方法上方
    @Pointcut("execution(void com.itheima.dao.BookDao.update1())")
    private void pt(){}

    //设置在切入点pt()的前面运行当前操作(前置通知)
     @Before("pt()")
    public void method(){
        System.out.println(System.currentTimeMillis());
    }
}

步骤3:运行程序 

步骤 4: 修改 MyAdvice 类,增强
因为定义的切入点中,被修改成 update , 所以 BookDao 中的 update 方法在执行的时候,就会被增
强,所以容器中的对象应该是目标对象的代理对象

@Component
//设置当前类为切面类类
@Aspect
public class MyAdvice {
    //设置切入点,要求配置在方法上方
    @Pointcut("execution(void com.itheima.dao.BookDao.update())")
    private void pt(){}

    //设置在切入点pt()的前面运行当前操作(前置通知)
     @Before("pt()")
    public void method(){
        System.out.println(System.currentTimeMillis());
    }
}
步骤 5: 运行程序

至此对于刚才的结论,我们就得到了验证,这块大家需要注意的是 :
不能直接打印对象,从上面两次结果中可以看出,直接打印对象走的是对象的 toString 方法,不管是不是代理对象打印的结果都是一样的,原因是内部对toString 方法进行了重写。

2.AOP核心概念

在上面介绍 AOP 的工作流程中,我们提到了两个核心概念,分别是 :
  • 目标对象(Target):原始功能去掉共性功能对应的类产生的对象,这种对象是无法直接完成最终工作的
  • 代理(Proxy):目标对象无法直接完成工作,需要对其进行功能回填,通过原始对象的代理对象实
上面这两个概念比较抽象,简单来说,目标对象就是要增强的类[ :BookServiceImpl ] 对应的对象,也叫原始对象,不能说它不能运行,只能说它在运行的过程中对于要增强的内容是缺失的。
SpringAOP 是在不改变原有设计 ( 代码 ) 的前提下对其进行增强的,它的底层采用的是代理模式实现
的,所以要对原始对象进行增强,就需要对原始对象创建代理对象,在代理对象中的方法把通知
[ :MyAdvice 中的 method 方法 ] 内容加进去,就实现了增强 , 这就是我们所说的代理 (Proxy)

四、AOP配置管理

1.AOP切入点表达式

前面的案例中,有涉及到如下内容:

 对于AOP中切入点表达式,我们总共会学习三个内容,分别是语法格式通配符书写技巧

1,语法格式

首先我们先要明确两个概念 :
  • 切入点:要进行增强的方法
  • 切入点表达式:要进行增强的方法的描述方式
对于切入点的描述,我们其实是有两中方式的,先来看下前面的例子

 描述方式一:执行com.itheima.dao包下的BookDao接口中的无参数update方法

 execution(void com.itheima.dao.BookDao.update())
描述方式二:执行 com.itheima.dao.impl 包下的 BookDaoImpl 类中的无参数 update 方法
 execution(void com.itheima.dao.impl.BookDaoImpl.update())
因为调用接口方法的时候最终运行的还是其实现类的方法,所以上面两种描述方式都是可以的。
对于切入点表达式的语法为 :
  • 切入点表达式标准格式:动作关键字(访问修饰符 返回值 包名./接口名.方法名(参数) 异常 名)
对于这个格式,我们不需要硬记,通过一个例子,理解它 :
execution(public User com.itheima.service.UserService.findById(int))
  • execution:动作关键字,描述切入点的行为动作,例如execution表示执行到指定切入点
  • public:访问修饰符,还可以是publicprivate等,可以省略
  • User:返回值,写返回值类型
  • com.itheima.service:包名,多级包使用点连接
  • UserService:/接口名称
  • findById:方法名
  • int:参数,直接写参数的类型,多个类型用逗号隔开
  • 异常名:方法定义中抛出指定异常,可以省略
切入点表达式就是要找到需要增强的方法,所以它就是对一个具体方法的描述,但是方法的定义会有很多,所以如果每一个方法对应一个切入点表达式,想想这块就会觉得将来编写起来会比较麻烦,有没有更简单的方式呢?
就需要用到下面所学习的通配符。

2.通配符

我们使用通配符描述切入点,主要的目的就是简化之前的配置,具体都有哪些通配符可以使用?

* : 单个独立的任意符号,可以独立出现,也可以作为前缀或者后缀的匹配符出现
execution(public * com.itheima.*.UserService.find*(*))
匹配 com.itheima 包下的任意包中的 UserService 类或接口中所有 find 开头的带有一个参数的
方法
.. :多个连续的任意符号,可以独立出现,常用于简化包名与参数的书写
execution(public User com..UserService.findById(..))
匹配 com 包下的任意包中的 UserService 类或接口中所有名称为 findById 的方法
+ :专用于匹配子类类型
execution(* *..*Service+.*(..))
这个使用率较低,描述子类的,咱们做 JavaEE 开发,继承机会就一次,使用都很慎重,所以很少
用它。 *Service+ ,表示所有以 Service结尾的接口的子类。        
接下来,我们把案例中使用到的切入点表达式来分析下 :
execution(void com.itheima.dao.BookDao.update())
匹配接口,能匹配到
execution(void com.itheima.dao.impl.BookDaoImpl.update())
匹配实现类,能匹配到
execution(* com.itheima.dao.impl.BookDaoImpl.update())
返回值任意,能匹配到
execution(* com.itheima.dao.impl.BookDaoImpl.update(*))
返回值任意,但是update方法必须要有一个参数,无法匹配,要想匹配需要在update接口和实现类添加
参数
execution(void com.*.*.*.*.update())
返回值为void,com包下的任意包三层包下的任意类的update方法,匹配到的是实现类,能匹配
execution(void com.*.*.*.update())
返回值为void,com包下的任意两层包下的任意类的update方法,匹配到的是接口,能匹配
execution(void *..update())
返回值为void,方法名是update的任意包下的任意类,能匹配
execution(* *..*(..))
匹配项目中任意类的任意方法,能匹配,但是不建议使用这种方式,影响范围广
execution(* *..u*(..))
匹配项目中任意包任意类下只要以u开头的方法,update方法能满足,能匹配
execution(* *..*e(..))
匹配项目中任意包任意类下只要以e结尾的方法,update和save方法能满足,能匹配
execution(void com..*())
返回值为void,com包下的任意包任意类任意方法,能匹配,*代表的是方法
execution(* com.itheima.*.*Service.find*(..))
将项目中所有业务层方法的以find开头的方法匹配
execution(* com.itheima.*.*Service.save*(..))
将项目中所有业务层方法的以save开头的方法匹配
                                                                        
后面两种更符合我们平常切入点表达式的编写规则

3.书写技巧

对于切入点表达式的编写其实是很灵活的,那么在编写的时候,有没有什么好的技巧让我们用用:

  • 所有代码按照标准规范开发,否则以下技巧全部失效
  • 描述切入点通常描述接口,而不描述实现类,如果描述到实现类,就出现紧耦合了
  • 访问控制修饰符针对接口开发均采用public描述(可省略访问控制修饰符描述
  • 返回值类型对于增删改类使用精准类型加速匹配,对于查询类使用*通配快速描述
  • 包名书写尽量不使用..匹配,效率过低,常用*做单个包描述匹配,或精准匹配
  • 接口名/类名书写名称与模块相关的采用*匹配,例如UserService书写成*Service,绑定业务层接口名
  • 方法名书写以动词进行精准匹配,名词采用匹配,例如getById书写成getBy,selectAll书写成 selectAll
  • 参数规则较为复杂,根据业务方法灵活调整
  • 通常不使用异常作为匹配规则

 2.AOP通知类型

前面的案例中,有涉及到如下内容:

 @Before("pt()")
它所代表的含义是将 通知 添加到 切入点 方法执行的 前面
除了这个注解外,还有没有其他的注解,换个问题就是除了可以在前面加,能不能在其他的地方加 ?

1.类型介绍

我们先来回顾下AOP通知:

AOP 通知描述了抽取的共性功能,根据共性功能抽取的位置不同,最终运行代码时要将其加入到合 理的位置
通知具体要添加到切入点的哪里 ?
共提供了 5 种通知类型 :
  • 前置通知
  • 后置通知
  • 环绕通知(重点)
  • 返回后通知(了解)
  • 抛出异常后通知(了解)
为了更好的理解这几种通知类型,我们来看一张图

(1) 前置通知,追加功能到方法执行前 , 类似于在代码 1 或者代码 2 添加内容
(2) 后置通知 , 追加功能到方法执行后 , 不管方法执行的过程中有没有抛出异常都会执行,类似于在代
5 添加内容
(3) 返回后通知 , 追加功能到方法执行后,只有方法正常执行结束后才进行 , 类似于在代码 3 添加内容,如果方法执行抛出异常,返回后通知将不会被添加
(4) 抛出异常后通知 , 追加功能到方法抛出异常后,只有方法执行出异常才进行 , 类似于在代码 4 添加内容,只有方法抛出异常后才会被添加
(5) 环绕通知 , 环绕通知功能比较强大,它可以追加功能到方法执行的前后,这也是比较常用的方式,它可以实现其他四种通知类型的功能,具体是如何实现的,需要我们往下学习。

2.环境准备

  • 创建一个Maven项目
  • pom.xml添加Spring依赖
  <dependencies>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.2.10.RELEASE</version>
    </dependency>
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.9.4</version>
    </dependency>
  </dependencies>

添加BookDaoBookDaoImpl 

public interface BookDao {
    public void update();

    public int select();
}
@Repository
public class BookDaoImpl implements BookDao {

    public void update(){
        System.out.println("book dao update is running ...");
    }

    public int select() {
        System.out.println("book dao select is running ...");

        return 100;
    }
}
创建 Spring 的配置类
@Configuration
@ComponentScan("com.itheima")
@EnableAspectJAutoProxy
public class SpringConfig {
}
创建通知类
@Component
 @Aspect
 public class MyAdvice {
 @Pointcut("execution(void com.itheima.dao.BookDao.update())")
 private void pt(){}

 public void before() {
 System.out.println("before advice ...");
}
 public void after() {
 System.out.println("after advice ...");
}
 public void around(){
 System.out.println("around before advice ...");
 System.out.println("around after advice ...");
}
 public void afterReturning() {
 System.out.println("afterReturning advice ...");
}
 public void afterThrowing() {
 System.out.println("afterThrowing advice ...");
}
}
编写 App 运行类
public class App {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(SpringConfig.class);
BookDao bookDao = ctx.getBean(BookDao.class);
bookDao.update();
}
}
最终创建好的项目结构如下 :

3.通知类型的使用

前置通知
修改 MyAdvice, before 方法上添加 @Before 注解
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Before("pt()")
//此处也可以写成 @Before("MyAdvice.pt()"),不建议
public void before() {
System.out.println("before advice ...");
}
}

 后置通知

@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Before("pt()")
public void before() {
System.out.println("before advice ...");
}
@After("pt()")
public void after() {
System.out.println("after advice ...");
}
}

环绕通知
基本使用
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Around("pt()")
public void around(){
System.out.println("around before advice ...");
System.out.println("around after advice ...");
}
}

 运行结果中,通知的内容打印出来,但是原始方法的内容却没有被执行。

因为环绕通知需要在原始方法的前后进行增强,所以环绕通知就必须要能对原始操作进行调用,具体如何实现?
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Around("pt()")
public void around(ProceedingJoinPoint pjp) throws Throwable{
System.out.println("around before advice ...");
//表示对原始操作的调用
pjp.proceed();
System.out.println("around after advice ...");
}
}
说明 : proceed() 为什么要抛出异常 ?

再次运行,程序可以看到原始方法已经被执行了

注意事项
(1) 原始方法有返回值的处理
修改 MyAdvice, BookDao 中的 select 方法添加环绕通知,
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Pointcut("execution(int com.itheima.dao.BookDao.select())")
private void pt2(){}
@Around("pt2()")
public void aroundSelect(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("around before advice ...");
//表示对原始操作的调用
pjp.proceed();
System.out.println("around after advice ...");
}
}
修改 App 类,调用 select 方法
public class App {
 public static void main(String[] args) {
 ApplicationContext ctx = new
AnnotationConfigApplicationContext(SpringConfig.class);
 BookDao bookDao = ctx.getBean(BookDao.class);
 int num = bookDao.select();
 System.out.println(num);
 }
 }

运行后会报错,错误内容为:

Exception in thread "main" org.springframework.aop.AopInvocationException:
Null return value from advice does not match primitive return type for:
public abstract int com.itheima.dao.BookDao.select() at
org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopP
roxy.java:226) at com.sun.proxy.$Proxy19.select(Unknown Source) at
com.itheima.App.main(App.java:12)
错误大概的意思是 : 空的返回不匹配原始方法的 int 返回
  • void就是返回Null
  • 原始方法就是BookDao下的select方法
所以如果我们使用环绕通知的话,要根据原始方法的返回值来设置环绕通知的返回值,具体解决方案为
public Object around(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("around before advice ...");
        //表示对原始操作的调用
        Object ret = pjp.proceed();

        System.out.println("around after advice ...");
        return ret;
    }
说明 :
为什么返回的是 Object 而不是 int 的主要原因是 Object 类型更通用。
在环绕通知中是可以对原始方法返回值就行修改的。
返回后通知
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Pointcut("execution(int com.itheima.dao.BookDao.select())")
private void pt2(){}
@AfterReturning("pt2()")
public void afterReturning() {
System.out.println("afterReturning advice ...");
}
}

注意: 返回后通知是需要在原始方法 select 正常执行后才会被执行,如果 select() 方法执行的过程
中出现了异常,那么返回后通知是不会被执行。后置通知是不管原始方法有没有抛出异常都会被执
行。
异常后通知
@Component
@Aspect
public class MyAdvice {
@Pointcut("execution(void com.itheima.dao.BookDao.update())")
private void pt(){}
@Pointcut("execution(int com.itheima.dao.BookDao.select())")
private void pt2(){}
@AfterReturning("pt2()")
public void afterThrowing() {
System.out.println("afterThrowing advice ...");
}
}

 

注意: 异常后通知是需要原始方法抛出异常,可以在 select() 方法中添加一行代码 int i = 1/0
可。如果没有抛异常,异常后通知将不会被执行。
学习完这 5 种通知类型,我们来思考下环绕通知是如何实现其他通知类型的功能的 ?
因为环绕通知是可以控制原始方法执行的,所以我们把增强的代码写在调用原始方法的不同位置就可以实现不同的通知类型的功能,如:

环绕通知注意事项
1. 环绕通知必须依赖形参 ProceedingJoinPoint 才能实现对原始方法的调用,进而实现原始方法
调用前后同时添加通知
2. 通知中如果未使用 ProceedingJoinPoint 对原始方法进行调用将跳过原始方法的执行
3. 对原始方法的调用可以不接收返回值,通知方法设置成 void 即可,如果接收返回值,最好设定为
Object 类型
4. 原始方法的返回值如果是 void 类型,通知方法的返回值类型可以设置成 void, 也可以设置成
Object
5. 由于无法预知原始方法运行后是否会抛出异常,因此环绕通知方法必须要处理 Throwable 异常
笔记来自: 黑马程序员SSM框架教程

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/498290.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python小姿势 - 1. Python的设计理念

Python的设计理念 Python的设计理念是“优雅”、“明确”、“简单”。 优雅&#xff1a;Python代码风格优美&#xff0c;语法简洁明了&#xff0c;代码可读性高&#xff0c;易于理解和维护。 明确&#xff1a;Python语言规范清晰&#xff0c;标准库丰富&#xff0c;可用于开发各…

第五章 作业(123)【编译原理】

第五章 作业【编译原理】 前言推荐第五章 作业123 随堂练习课前热身04-17随堂练习04-17课前热身04-24 最后 前言 2023-5-3 22:12:46 以下内容源自《【编译原理】》 仅供学习交流使用 推荐 第四章 作业&#xff08;123&#xff09;【编译原理】 第五章 作业 1 1.令文法G为…

医生的百科词条怎么创建?医生的百科词条创建技巧值得你收藏

医生是医学领域中的专业人员&#xff0c;主要负责诊断、治疗和预防疾病。在现代社会中&#xff0c;医生的角色越来越重要&#xff0c;是社会中不可或缺的职业之一。 而随着互联网的发展&#xff0c;百度百科已成为人们获取信息的重要途径&#xff0c;医生想要提高自己的知名度和…

SpringBoot项目简单入门

一、创建项目 1、选择Spring Initializr 2、为了提高项目构建效率&#xff0c;可以尝试修改阿里脚手架&#xff0c;地址如下&#xff1a; https://start.aliyun.com 3、点击下一步 4、选择Web与spring Web&#xff0c;然后点击完成开始项目构建 5、项目构建完成如图所示 二、…

Linux基础知识—Linux

文章目录 1.认识Linux2.常见命令2.1ls2.2pwd2.3cd2.4touch2.5mkdir2.6rm2.7cp2.8mv2.9man2.10date2.11grep2.12ps2.13netstat 3.文件内容的操作3.1cat3.2vim3.3less3.4head3.5tail3.6管道|3.7重定向 4.管理软件4.1yum&#xff08;在线的方式管理&#xff09;4.2rpm&#xff08;…

OnlineJudge-负载均衡式在线OJ

关于个人项目是在找实习以及参加秋招非常重要的简历内容&#xff0c;今天博主来介绍一下自己的一个项目。 开发环境&#xff1a;CentOS7、Makefile、g、vscode、MySQL Workbench 所用技术&#xff1a;C STL 标准库、Boost 准标准库(字符串切割)、cpp-httplib 第三方开源网络库 …

数据结构(C语言):两个字符串比较大小

一、一个小插曲 在写这篇文章之前&#xff0c;作者想先和大家分享一个小故事。如果你不想看这个小故事的话&#xff0c;可以直接跳到第二点哦。 为了锻炼自己的编码能力&#xff0c;平时作业和实验题的代码我都是不看书、不看老师的PPT&#xff0c;按照自己的思路一行一行敲出…

【STM32CubeMX】F103RTC时钟

前言 本文记录了我学习STM32CubeMX的过程&#xff0c;方便以后回忆。我们使用的开发板是基于STM32F103C6T6的。本章记录了RTC时钟的基础配置。下文调试时用到的串口来查看&#xff0c;不过串口的配置省略了。 步骤 实验目标&#xff1a;基于RTC时钟&#xff0c;查看它的秒计时…

Mac电脑配置李沐深度学习环境[pytorch版本]使用vscode

文章目录 第一步 M1芯片安装Pytorch环境安装Miniforge创建虚拟环境安装Pytorch 第二步 下载李沐Jupyter文件第三步 配置vscode参考 第一步 M1芯片安装Pytorch环境 安装Miniforge Mac打开终端&#xff08;Mac电脑如何启动终端&#xff1f;打开启动台&#xff0c;搜索终端即可&…

网络安全合规-数据分类分级标准汇编

今天主要学习讲解的是网络安全合规-数据分类分级标准汇编。 作为数据安全治理的前期首要工作-分类分级&#xff0c;而分类分级的开展工作又是根据相关标准开展的&#xff0c;建立数据安全防护体系的第一步就是梳理数据资产进行分类分级。只有做好分类分级工作&#xff0c;对不同…

迈向多模态AGI之开放世界目标检测 | 人工智能

作者&#xff1a;王斌 谢春宇 冷大炜 引言 目标检测是计算机视觉中的一个非常重要的基础任务&#xff0c;与常见的的图像分类/识别任务不同&#xff0c;目标检测需要模型在给出目标的类别之上&#xff0c;进一步给出目标的位置和大小信息&#xff0c;在CV三大任务&#xff08;识…

GIMP制作艺术字技巧

GIMP下载官网 https://www.gimp.org/downloads/ 我使用的版本 2.10.32 字体下载 https://ziyouziti.com/index-index-all.html 下载解压之后会有otf、ttf等字体文件&#xff0c;需要拷贝到gimp当前用户目录 C:\Users\用户名\AppData\Roaming\GIMP\2.10\fonts GIMP绘制字…

分布式医疗云平台【项目简介、适合对象、技术选型、项目的核心功能模块 、模块设计及功能演示】(一)-全面详解(学习总结---从入门到深化)

目录 分布式医疗云平台 一、项目简介 二、适合对象 三、技术选型 四、项目的核心功能模块 五、项目特色 六、模块设计及功能演示 分布式医疗云平台 一、项目简介 分布式医疗云平台系统是以完整的基层医疗机构信息化解决方案为出发点&#xff0c;打造链接诊所、医生、…

【P7】JMeter 计数器

&#xff08;1&#xff09;、测试计划右键 <<< 添加 <<< 配置元件 <<< 计数器 Starting value&#xff1a;1 递增&#xff1a;1 Maximum value&#xff1a;9999 数字格式&#xff1a;var_0000 引用名称&#xff1a;var &#xff08;2&#xf…

web集群,部署jpress应用

1.静态网页与动态网页的区别 静态网页&#xff1a; &#xff08;1&#xff09;请求响应信息&#xff0c;发送给客户端进行处理&#xff0c;由浏览器进行解析&#xff0c;显示页面称为静态页面。在网站设计中&#xff0c;纯粹html格式的网页&#xff08;包含图片&#xff0c;视…

學習日記,java与写题目

开篇来个每日一题 1419. 数青蛙 难度中等185收藏分享切换为英文接收动态反馈 给你一个字符串 croakOfFrogs&#xff0c;它表示不同青蛙发出的蛙鸣声&#xff08;字符串 "croak" &#xff09;的组合。由于同一时间可以有多只青蛙呱呱作响&#xff0c;所以 croakOfF…

冯诺依曼+OS+进程+进程状态

索引 一.冯诺依曼理解二.OS 进程的初步认识1.什么是进程&#xff1f;2.如何查看进程3.父进程与子进程4.进程状态1.S阻塞态R运行态2.D阻塞&#xff08;不可中断&#xff09;3.Z僵尸状态andX死亡状态4.孤儿进程5&#xff0c;进程死亡之后OS做了什么五.状态总结&#xff1a; 一.冯…

数据库系统工程师 —— 第六章 数据库技术基础

文章目录 &#x1f4c2; 第六章、数据库技术基础 &#x1f4c1; 6.1 基本概念 &#x1f4d6; 6.1.1 数据库与数据库管理系统 &#x1f4d6; 6.1.2 数据库技术的发展 &#x1f4d6; 6.1.3 DBMS的功能和特点 &#x1f4d6; 6.1.4 数据库系统的体系结构 &#x1f4d6; 6.1.5 数据库…

极致鸿蒙——华为MatePad系列安装AidLux,一个自带vscode的Python编译环境

看着刚刚人入手的华为鸿蒙系统MatePad11平板&#xff0c;是如此的美轮美奂&#xff0c;但是总感觉少了点什么&#xff0c;少了点什么呢&#xff1f;是编程环境&#xff0c;我爱MatePad&#xff0c;也爱编程&#xff0c;那如果可以在MatePad上编程&#xff0c;会发生什么&#x…

Ae:3D 变换小工具与 3D 轴模式

◆ ◆ ◆ 3D 变换小工具 3D 变换小工具 3D Gizmo是用不同颜色标志的直观的调整工具&#xff0c;可用来缩放、定位和旋转 3D 图层、摄像机图层和灯光图层。 如上图所示&#xff0c;不同的颜色表示不同的轴。红色表示 X 轴&#xff0c;绿色表示 Y 轴&#xff0c;蓝色表示 Z 轴。…