[day14 SpringBoot 事务管理+AOP]

本文详细介绍了事务的基本概念、MySQL中的事务操作、Spring中事务的管理和异常处理,以及AOP(面向切面编程)的原理、通知类型和切点表达式。涵盖了事务开启、提交、回滚,以及如何在Spring中使用@Transactional注解进行事务控制和AOP实战示例。
摘要由CSDN通过智能技术生成

1. 事务是什么?在MySQL中是怎么去开启,提交,回滚事务的?

1.1 事务是什么

事务是一组操作的集合,它是一个不可分割的工作单位,事务会把所有的操作作为一个整体一起向系统提交或撤销操作请求,即这些操作要么同时成功,要么同时失败

1.2 事务操作

开启事务:begin /start transaction ;   一组操作开始前,开启事务

提交事务:   commit;   全部成功后提交事务

回滚事务:   rollback;   中间有任何一个子操作出现异常,回滚事务

2. 在Spring中怎么管理事务,用什么注解,在哪个层次进行使用可以有效管理事务

 注解:@Transactional

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

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

可以通过如下配置,查看详细的事务管理日志:

logging:
  level:
    org.springframework.jdbc.support.JdbcTransactionManager: debug

3. 事务默认识别的异常是运行时异常,如果想要识别到编译时异常从而实现回滚,该怎么操作?

rollbackFor属性可以控制出现何种异常类型,回滚事务。默认情况下,只有出现RuntimeException才回滚异常。而如果出现编译时异常,则不回滚。

一般情况下,如果业务代码有编译时异常,将其转换为运行时异常,再抛出,这样既方便使用,又不至于让事务失效。当然,如果全部抛出RuntimeException不利于调错,故而可以自定义运行时异常,并抛出自定义异常。

public class CustomerException extends RuntimeException{
    public CustomerException() {
    }
    public CustomerException(String message) {
        super(message);
    }
}
@Transactional
@Override
public void delete(Integer id) throws Exception {
    //1. 删除部门
    deptMapper.delete(id);
    
    try {
        InputStream in = new FileInputStream("E:/1.txt");
    } catch (Exception e) {
        throw new Exception("出错了");
    }
	
    //2. 根据部门id, 删除部门下的员工信息
    empMapper.deleteByDeptId(id);
}

4. 事务的传播方式有哪些,含义是怎样?

4.1 事务传播行为

指的就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行事务控制。

4.2 事务传播方式

属性值

含义

说明

REQUIRE

【默认值】需要事务,有则加入,无则创建新事务

-

REQUIRES_NEW

需要新事务,无论有无,总是创建新事务

-

SUPPORTS

支持事务,有则加入,无则在独立的连接中运行 SQL

结合 Hibernate、JPA  时有用,配在查询方法上

NOT_SUPPORTED

不支持事务,不加入,在独立的连接中运行 SQL

-

MANDATORY

必须有事务,否则抛异常

-

NEVER

必须没事务,否则抛异常

-

NESTED

嵌套事务

仅对 DataSourceTransactionManager  有效

我们主需要掌握前两个:REQUIRE以及REQUIRES_NEW,其他很少用到,无需掌握

5. AOP叫做什么,有什么作用?举一个5岁孩子就能理解的例子

  • AOP:Aspect Oriented Programming(面向切面编程),它的核心思想是将重复的逻辑剥离出来,在不修改原始逻辑的基础上对原始功能进行增强。
  • 优势:无侵入、减少重复代码、提高开发效率、维护方便

孩子能理解的例子:

我们把出去玩分为几个步骤:

1. 约好朋友一起出去玩

2.  穿好衣服做好准备

3. 一起玩

4. 玩完了回家

其中,每次出去万都要执行1、2、4步,所以将他们剥离出来,以后不管去哪玩都可以直接用了

6. 如果要完成一个AOP的入门案例,需要有哪些步骤?

6.1. 去pom文件引入依赖

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

6.2. 定义类抽取公共代码(执行耗时统计操作)

@Slf4j
public class TimeAspect {
	
    public void recordTime() throws Throwable {
        long begin = System.currentTimeMillis();
        
        //调用原始操作
        
        
        long end = System.currentTimeMillis();
        log.info("执行耗时 : {} ms", (end-begin));
    }
	
}

6.3. 标识当前类是一个AOP类,并被Spring容器管理

@Aspect:标识当前类是一个AOP类

@Component:声明该类是spring的IOC容器中的bean对象

@Component
@Aspect
@Slf4j
public class TimeAspect {
	
    public void recordTime() throws Throwable {
        long begin = System.currentTimeMillis();
        
        //调用原始操作
        
        
        long end = System.currentTimeMillis();
        log.info("执行耗时 : {} ms", (end-begin));
    }
	
}

6.4. 配置公共代码作用于哪些目标方法

@Around: 表示环绕通知,可以在目标方法执行前后执行一些公共代码

* 表示通配符,代表任意

.. 表示参数通配符,代表任意参数

@Component
@Aspect
@Slf4j
public class TimeAspect {
	
    @Around("execution(* com.itheima.service.impl.DeptServiceImpl.*(..))")
    public void recordTime() throws Throwable {
        long begin = System.currentTimeMillis();
        
        //调用原始操作
        
        
        long end = System.currentTimeMillis();
        log.info("执行耗时 : {} ms", (end-begin));
    }
	
}

6.5. 执行目标方法

@Component
@Aspect
@Slf4j
public class TimeAspect {
	
    @Around("execution(* com.itheima.service.impl.DeptServiceImpl.*(..))")
    public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
        long begin = System.currentTimeMillis();
        //调用原始操作
        Object result = joinPoint.proceed();
        long end = System.currentTimeMillis();
        log.info("执行耗时 : {} ms", (end-begin));
        return result;
    }
    
}

6.6. 测试

7.  AOP的概念理解:连接点,通知,切入点,切面 ?

  • 连接点:JoinPoint,可以被AOP控制的方法执行(包含方法信息)
  • 通知:Advice ,重复逻辑代码
  • 切入点:PointCut ,匹配连接点的条件
  • 切面:Aspect,通知+切点

8. 通知有哪些类型?

@Around:此注解标注的通知方法在目标方法前、后都被执行

@Before:此注解标注的通知方法在目标方法前被执行

@After :此注解标注的通知方法在目标方法后被执行,无论是否有异常

@AfterReturning : 此注解标注的通知方法在目标方法后被执行,有异常不会执行

@AfterThrowing : 此注解标注的通知方法发生异常后执行

其中 @Around 需要自己调用 ProceedingJoinPoint.proceed() 来让目标方法执行,其他通知不需要考虑目标方法执行

9. 如果有不同切面的通知,增强相同的方法,执行顺序是怎样的?

9.1. 关于通知的描述

如果还有下一个通知,则调用下一个通知

如果没有下一个通知,则调用目标

9.2. 通知的执行顺序

默认按照 bean 的名称字母排序

@Order(数字) 加在切面类上来控制顺序

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

        9.2.2. 目标后的通知方法:数字小后执行

定义TimeAspectAimeAspect 切面类,测试执行顺序,默认按照切面类的名称字母排序。此例中AimeAspect  比 TimeAspect 字母顺序排名靠前,故此,AimeAspect 先执行。

10. 切点表达式有几种,具体怎么定义?

10.1. execution(返回值类型 包名.类名.方法名(参数类型)

* 可以通配任意返回值类型、包名、类名、方法名、或任意类型的一个参数

.. 可以通配任意层级的包、或任意类型、任意个数的参数

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

10.2. @annotation() 根据注解匹配

args() 根据方法参数匹配

@annotation(com.itheima.anno.Log)

11. 如果有多个通知的切点表达式一样,怎么抽取?

通过@PointCut注解,可以抽取一个切入点表达式,然后再其他的地方我们就可以通过类似于  方法调用 的形式来引用该切入点表达式。

    @Pointcut("execution(* com.itheima.service.impl.DeptServiceImpl.*(..))")
    public void pt(){}

    @Around("pt()")
    public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
        long begin = System.currentTimeMillis();
        //调用原始操作
        Object result = joinPoint.proceed();
        long end = System.currentTimeMillis();
        log.info("执行耗时 : {} ms", (end-begin));
        return result;
    }

12. 连接点是什么可以获取哪些信息?

连接点简单理解就是目标方法,在Spring 中用 JoinPoint 抽象了连接点,用它可以获得方法执行时的相关信息,如方法名、方法参数类型、方法实际参数等等

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

关于如何连接点获取的信息,见下方代码

@Slf4j
@Aspect
@Component
public class MyAspect1 {

    @Pointcut("execution(* com.itheima.service.impl.*.*(..)) && @annotation(com.itheima.anno.Log)")
    public void pt(){}

    @Before("pt()")
    public void before(JoinPoint joinPoint){

        log.info("方法名: "+joinPoint.getSignature().getName());
        log.info("类名: "+joinPoint.getTarget().getClass().getName());
        log.info("参数: "+Arrays.asList(joinPoint.getArgs()).toString());

        log.info("before...1");
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值