Spring如何进行事务管理?什么是面向切面编程?

d874b92250061004831ce4f7b89820cd.gif

喜欢就点击上方关注我们吧!

4cded0eb3634b77115c91874c088ea7f.png

本篇将带你快速了解Spring事务管理以及面向切面编程(AOP)相关知识。

5a68204963d13715dbd522b3a0e9c20a.jpeg

一、事务

1、概述

1)事务是一组操作的集合,是一个不可分割的工作单位,这些操作要么同时成功,要么同时失败。

2)事务作用︰在数据层保障一系列的数据库操作同成功同失败

3)Spring事务作用︰在数据层或业务层保障一系列的数据库操作同成功同失败

2、事务操作

1)开启事务: start transaction / begin;

一组操作开始前,开启事务

2)提交事务: commit;

一组操作全部成功后,提交事务

3)回滚事务: rollback;

中间任何一个操作出现异常,回滚事务

3、事务管理

1)注解: @Transactional

2)位置: 业务(service)层的方法上、类上、接口上

3)作用: 将当前方法交给spring进行事务管理,方法执行前,开启事务;成功执行完毕,提交事务;出现异常,回滚事务

2e2fdd0c24ce68b275bb52ec9e6d1e21.jpeg

4)开启事务管理日志

#开启事务管理日志
logging:
  level:
    org.springframework.jdbc.support.JdbcTransactionManager: debug

5)几点注意

Spring注解式事务通常添加在业务层接口中而不会添加到业务层实现类中,降低耦合

注解式事务可以添加到业务方法上表示当前方法开启事务,也可以添加到接口上表示当前接口所有方法开启事务

注:Mybatis框架使用的是JDBC事务

4、相关属性

4.1 rollbackFor

rollbackFor属性用于控制出现何种异常类型,回滚事务。

默认情况下,只有出现RuntimeException才回滚异常。

如下配置所有的异常都会回滚异常

@Transactional(rollbackFor = Exception.class)
    @Override
    public void delete(Integer id) throws Exception{
        deptMapper.deteById(id); // 删除部门
        if(true){ // 永远成立,会抛出如下异常(非运行时异常)
            throw new Exception("出错了~");
        }
        empMapper.deleteByDeptId(id); // 删除部门下的员工
    }
4.2 传播行为

重点掌握前两个即可

2ae815071f8627db860639b3069957cc.png

REQUIRED: 大部分情况下都是用该传播行为。

REQUIRES_NEW: 当不希望事务之间相互影响时,可以使用该传播行为。如:下订单前需要记录日志,不论订单保存成功与否,都需要保证日志记录能够记录成功。

517c07c43eb03b0081c1c5daf7254dad.png

二、AOP

1、概述

AOP(Aspect Oriented Programming),面向切面编程,一种编程范式,指导开发者如何组织程序结构。

注意将我们之前学的面向对象区分开,OOP(Object oriented Programming),面向对象编程

AOP的本质:代理模式

作用:在不惊动原始设计的基础上为其进行功能增强

Spring理念:无入侵式/无侵入式

2、入门程序

以统计各个业务层方法执行耗时为例。

1)导入依赖: 在pom.xml中引入AOP依赖

<!-- 引入AOP依赖-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

2)编写AOP程序

e7e9b2cdae6a58751dc353d0cb0fed1d.png

AOP的应用场景:1、记录操作日志 2、权限控制 3、事务管理,等等

AOP优点:1、代码无侵入 2、减少重复代码 3、提高开发效率 4、方便维护

3、核心概念

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

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

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

在SpringAOP中,一个切入点可以只描述一个具体方法,也可以匹配多个方法

一个具体方法:如 com.itweb.dao包下的BookDao接口中的无形参无返回值的save方法

匹配多个方法:如所有的save方法,所有的get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法

1a9a6dc5f75a16a7be98d5036c2f6d6e.jpeg

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

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

通知类:定义通知的类

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

5)目标对象(Target):通知所应用的对象

27ee4235927d5c487aef649d8ae027e6.jpeg

6)AOP执行流程

32544b6ca2fca29a3a3ecda0f2f77a7c.jpeg

4、进阶

4.1 AOP-通知类型

1)AOP通知描述了抽取的共性功能,根据共性功能抽取的位置不同,最终运行代码时要将其加入到合理的位置

MyBatis的动态SQL也用到了共性抽取理念。

2)5种通知类型

1、前置通知(@Before):该注解标注的通知方法在目标方法前被执行

2、后置通知(@After):此注解标注的通知方法在目标方法后被执行,无论是否有异常都会执行。类似与异常处理的 finally语句

3、环绕通知(@Around--重点:此注解标注的通知方法在目标方法前、后都被执行

4、返回后通知(@AfterReturning)--了解

5、抛出异常后通知(@AfterThrowing)--了解

4.2 Around

注意事项

1)环绕通知必须依赖形参ProceedingJoinPoint才能实现对原始方法的调用,进而实现原始方法调用前后同时添加通知

2)通知中如果未使用ProceedingJoinPoint对原始方法进行调用将跳过原始方法的执行

3)对原始方法的调用可以不接收返回值,通知方法设置成void即可,如果接收返回值,必须设定为Object类型

4)原始方法的返回值如果是void类型,通知方法的返回值类型可以设置成void,也可以设置成Object

5)由于无法预知原始方法运行后是否会抛出异常,因此环绕通知方法必须抛出Throwable对象

示例

@Around("pt()")
    public around(ProceedingJoinPoint pjp) throws Throwable { // 抛出异常
        System.out.println("around before advice...");
        Object ret = pjp.proceed(); // 调用原始操作
        System.out.println("around after advice...");
        return ret;
    }
4.3 通知顺序(了解)

1)背景

当有多个切面的切入点都匹配到了目标方法,目标方法运行时,多个通知方法都会被执行。那么执行顺序是怎么样的?

2)顺序规则

不同切面类中,默认按照切面类的类名字母排序:

目标方法前的通知方法: 字母排名靠前的先执行

目标方法后的通知方法: 字母排名靠前的后执行

3)改变顺序

法1:手动更改类名(繁琐)

法2:@Order(数字),加在切面类上来控制顺序(推荐)

目标方法前的通知方法: 数字小的先执行 目标方法后的通知方法: 数字小的后执行

4.4 常见切入点表达式

所谓切入点表达式,就是描述切入点方法的一种表达式。其主要用来决定项目中的哪些方法需要加入通知。

1)execution(最常用)

语法形式:

execution(访问修饰符 返回值 包名.类名.方法名(方法参数) throws 异常)

其中访问修饰符(如 public,protected),包名.类名.throws 异常,这些可以省略,但是不建议将包名.类名.省略,防止匹配范围过大而影响匹配效率。

通配符:可用于描述切入点

590d6870b467b03d5d89382ad442caaf.png

注:可以使用且(&&)、或(||)、非(!)来组合比较复杂的切入点表达式。

书写规范:

所有业务方法名在命名时尽量规范,方便切入点表达式快速匹配。如: 查询类方法都是find 开头,更新类方法都是update开头。

描述切入点方法通常基于接口描述,而不是直接描述实现类,增强拓展性(当接口的实现类改变时无需改变切入点方法)

在满足业务需要的前提下,尽量缩小切入点的匹配范围,提高匹配效率。如: 包名匹配尽量使用 *匹配单个包,而不使用..。

2)@annotation

annotation切入点表达式,用于匹配标识有特定注解的方法。

用法:@annotation(注解全类名),用法相比 execution更加灵活

@annotation(com.itweb.anno.Log)
@Before ("@annotation (com.itweb.anno.Log)")
public void before (){
    log.info ("before ....");
}
4.5 连接点

在Spring中用JoinPoint抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等。

616672408bbf0bc872ae8ac12b7ee603.png

对于@Around通知,获取连接点信息只能使ProceedingJoinPoint

对于其他四种通知,获取连接点信息只能使用JoinPoint ,它是 ProceedingJoinPoint的父类型

6604f6a0c3c2b9dc6cc18e86bffa7248.png

63393066fda46af087a41431300837c3.gif

喜欢就点击上方关注我们吧!

45a8a2a375abd0812106695005232e93.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

码农后端

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值