当前位置: 首页 > news >正文

AOP面向切面编程思想。

目录

一、AOP工作流程

1、基本概念

2、AOP工作流程 

二、AOP核心配置

1、AOP切入点表达式

2、AOP通知类型

三、AOP通知获取数据

1、获取参数

2、获取返回值

3、获取异常 

四、AOP事务管理

1、Spring事务简介

2、Spring事务角色 

3、事务属性


一、AOP工作流程

1、基本概念

▶ 什么是AOP?

 ● AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程序结构。 通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。
  ● OOP(Object Oriented Programming)面向对象编程

我们都知道OOP是一种编程思想,那么AOP也是一种编程思想,编程思想主要的内容就是指导程序员该如何编写程序,所以它们两个是不同的`编程范式`。

▶ AOP作用

    AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

▶ 核心概念

● 连接点(JoinPoint):程序执行过程中的任意位置,粒度为执行方法、抛出异常、设置变量等,在SpringAOP中,理解为方法的执行。
● 切入点(Pointcut) : 匹配连接点的式子
  ○ 在SpringAOP中,一个切入点可以描述一个具体方法,也可也匹配多个方法:
    ① 一个具体的方法 : 如com.itheima.dao包下的BookDao接口中的无形参无返回值的save方法
    ② 匹配多个方法 : 所有的save方法,所有的get开头的方法,所有以Dao结尾的接口中的任意方法,所有带有一个参数的方法
  ○ 连接点范围要比切入点范围大,是切入点的方法也一定是连接点,但是是连接点的方法就不一定要被增强,所以可能不是切入点。
● 通知(Advice): 在切入点处执行的操作,也就是共性功能。在SpringAOP中,功能最终以方法的形式呈现。
● 通知类:定义通知的类
● 切面(Aspect): 描述通知与切入点的对应关系。

(1) Spring的AOP是对一个类的方法在不进行任何修改的前提下实现增强。对于上面的图示中中有`save`,`update`,`delete`和`select`方法,这些方法我们给起了一个名字叫连接点

(2) 在四个方法中,`update`和`delete`只有打印没有计算万次执行消耗时间,但是在运行的时候已经有该功能,那也就是说`update`和`delete`方法都已经被增强,所以对于需要增强的方法我们给起了一个名字叫切入点

(3) 执行update和delete方法的时候都被添加了一个计算万次执行消耗时间的功能,将这个功能抽取到一个方法中,换句话说就是存放共性功能的方法,我们给起了个名字叫通知

(4) 通知是要增强的内容,会有多个,切入点是需要被增强的方法,也会有多个,那哪个切入点需要添加哪个通知,就需要提前将它们之间的关系描述清楚,那么对于通知和切入点之间的关系描述,我们给起了个名字叫切面

(5) 通知是一个方法,方法不能独立存在需要被写在一个类中,这个类我们也给起了个名字叫通知类

▶ 知识点

@EnableAspectJAutoProxy

@Aspect

@Pointcut   

@Before

▶ 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 : 定义接口与实现类

 前面写的业务层service的代码类似。

▷ 步骤3 : 定义通知类和通知

通知就是将共性功能抽取出来后形成的方法,共性功能指的就是当前系统时间的打印。类名和方法名没有要求,可以任意。

public class MyAdvice {public void method(){System.out.println(System.currentTimeMillis());}
}

▷ 步骤4 : 定义切入点

public class MyAdvice {@Pointcut("execution(void com.itheima.dao.BookDao.update())")private void pt(){}public void method(){System.out.println(System.currentTimeMillis());}
}

注意:切入点定义依托一个不具有实际意义的方法进行,即无参数、无返回值、方法体无实际逻辑。 

▷ 步骤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编程成功。

2、AOP工作流程 

▶ 流程1 : Spring容器启动

● 容器启动就需要去加载bean,哪些类需要被加载呢?
● 需要被增强的类,如:BookServiceImpl
● 通知类,如:MyAdvice
● 注意此时bean对象还没有创建成功

▶ 流程2 : 读取所有切面配置中的切入点

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

▶ 流程3 : 初始化bean

判定bean对应的类中的方法是否匹配到任意切入点

▶ 注意第1步在容器启动的时候,bean对象还没有被创建成功。

▶ 要被实例化bean对象的类中的方法和切入点进行匹配

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

▶ 流程4 : 获取bean执行方法

 ● 获取的bean是原始对象时,调用方法并执行,完成操作
 ● 获取的bean是代理对象时,根据代理对象的运行模式运行原始方法与增强的内容,完成操作

▶ 容器是否为代理对象

 结论:

 ● 如果目标对象中的方法会被增强,那么容器中将存入的是目标对象的代理对象
 ● 如果目标对象中的方法不被增强,那么容器中将存入的是目标对象本身。

▶ AOP核心概念

 ● 目标对象(Target):原始功能去掉共性功能对应的类产生的对象,这种对象是无法直接完成最终工作的
 ● 代理(Proxy):目标对象无法直接完成工作,需要对其进行功能回填,通过原始对象的代理对象实现

▷ 上面这两个概念比较抽象,简单来说:

    目标对象就是要增强的类[如:BookServiceImpl类]对应的对象,也叫原始对象,不能说它不能运行,只能说它在运行的过程中对于要增强的内容是缺失的。

    SpringAOP是在不改变原有设计(代码)的前提下对其进行增强的,它的底层采用的是代理模式实现的,所以要对原始对象进行增强,就需要对原始对象创建代理对象,在代理对象中的方法把通知[如:MyAdvice中的method方法]内容加进去,就实现了增强,这就是我们所说的代理(Proxy)。

二、AOP核心配置

 1、AOP切入点表达式

● 示例:

▶ 语法格式

▷ 明确两个概念:

 ● 切入点 : 要进行增强的方法
 ● 切入点表达式 : 要进行增强的方法的描述方式

▷ 对于切入点的描述,有两中方式:

 ● 描述方式一:执行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 : 访问修饰符,还可以是public,private等,可以省略
  • User:返回值,写返回值类型
  • com.itheima.service :包名,多级包使用点连接
  • UserService : 类/接口名称
  • findById:方法名
  • int : 参数,直接写参数的类型,多个类型用逗号隔开
  • 异常名:方法定义中抛出指定异常,可以省略

   切入点表达式就是要找到需要增强的方法,所以它就是对一个具体方法的描述,但是方法的定义会有很多,所以如果每一个方法对应一个切入点表达式。

▶ 通配符

▷  `* `: 单个独立的任意符号,可以独立出现,也可以作为前缀或者后缀的匹配符出现

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开头的方法匹配 

▶ 书写技巧

所有代码按照标准规范开发,否则以下技巧全部失效:

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

2、AOP通知类型

▶ 类型介绍

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

▷ 共提供了5种通知类型:

  • 前置通知
  • 后置通知
  • 环绕通知(重点)
  • 返回后通知(了解)
  • 抛出异常后通知(了解)

▷ 理解

(1) 前置通知,追加功能到方法执行前, 类似于在代码1或者代码2添加内容

(2) 后置通知, 追加功能到方法执行后, 不管方法执行的过程中有没有抛出异常都会执行,类似于在代码5添加内容

(3) 返回后通知, 追加功能到方法执行后,只有方法正常执行结束后才进行, 类似于在代码3添加内容,如果方法执行抛出异常,返回后通知将不会被添加

(4) 抛出异常后通知, 追加功能到方法抛出异常后,只有方法执行出异常才进行,类似于在代码4添加内容,只有方法抛出异常后才会被添加

(5) 环绕通知, 环绕通知功能比较强大,它可以追加功能到方法执行的前后,这也是比较常用的方式,它可以实现其他四种通知类型的功能。

▶ 使用时需要添加的依赖

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>

▶ 通知类型的使用

▷ 前置通知

  修改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(){}@Pointcut("execution(int com.itheima.dao.BookDao.select())")private void pt2(){}@Around("pt2()")public Object aroundSelect(ProceedingJoinPoint pjp) throws Throwable {System.out.println("around before advice ...");//表示对原始操作的调用Object ret = pjp.proceed();System.out.println("around after advice ...");return ret;}
}

▷ 返回后通知

@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`即可。如果没有抛异常,异常后通知将不会被执行。

▶ 环绕通知注意事项

   环绕通知是如何实现其他通知类型的功能的? 因为环绕通知是可以控制原始方法执行的,所以我们把增强的代码写在调用原始方法的不同位置就可以实现不同的通知类型的功能,如:

注意:

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

▶ 通知类型总结

@After

@AfterReturning 

@AfterThrowing  

@Around

三、AOP通知获取数据

1、获取参数

▶ 非环绕通知获取方式

  在方法上添加JoinPoint,通过JoinPoint来获取参数

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@Before("pt()")public void before(JoinPoint jp) Object[] args = jp.getArgs();System.out.println(Arrays.toString(args));System.out.println("before advice ..." );}//...其他的略
}

参数可以是多个:

@Repository
public class BookDaoImpl implements BookDao {public String findName(int id,String password) {System.out.println("id:"+id);return "itcast";}
}

注意:使用JoinPoint的方式获取参数适用于`前置`、`后置`、`返回后`、`抛出异常后`通知。

▶ 环绕通知获取方式

    环绕通知使用的是ProceedingJoinPoint,因为ProceedingJoinPoint是JoinPoint类的子类,所以对于ProceedingJoinPoint类中应该也会有对应的`getArgs()`方法,我们去验证下:

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@Around("pt()")public Object around(ProceedingJoinPoint pjp)throws Throwable {Object[] args = pjp.getArgs();System.out.println(Arrays.toString(args));Object ret = pjp.proceed();return ret;}//其他的略
}

注意:

● pjp.proceed()方法是有两个构造方法,分别是:

  ○ 调用无参数的proceed,当原始方法有参数,会在调用的过程中自动传入参数

  ○ 所以调用这两个方法的任意一个都可以完成功能

  ○ 但是当需要修改原始方法的参数时,就只能采用带有参数的方法,如下:

@Component
@Aspect
public class MyAdvice {Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@Around("pt()")public Object around(ProceedingJoinPoint pjp) throws Throwable{Object[] args = pjp.getArgs();System.out.println(Arrays.toString(args));args[0] = 666;Object ret = pjp.proceed(args);return ret;}//其他的略
}

    有了这个特性后,我们就可以在环绕通知中对原始方法的参数进行拦截过滤,避免由于参数的问题导致程序无法正确运行,保证代码的健壮性。

2、获取返回值

▶ 环绕通知获取返回值

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@Around("pt()")public Object around(ProceedingJoinPoint pjp) throws Throwable{Object[] args = pjp.getArgs();System.out.println(Arrays.toString(args));args[0] = 666;Object ret = pjp.proceed(args);return ret;}//其他的略
}

上述代码中,`ret`就是方法的返回值,我们是可以直接获取,不但可以获取,如果需要还可以进行修改。

▶ 返回后通知获取返回值

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@AfterReturning(value = "pt()",returning = "ret")public void afterReturning(Object ret) {System.out.println("afterReturning advice ..."+ret);}//其他的略
}

注意:

(1)参数名的问题

(2)afterReturning方法参数类型的问题

   参数类型可以写成String,但是为了能匹配更多的参数类型,建议写成Object类型

(3)afterReturning方法参数的顺序问题

3、获取异常 

▶ 环绕通知获取异常

这块比较简单,以前我们是抛出异常,现在只需要将异常捕获,就可以获取到原始方法的异常信息了

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@Around("pt()")public Object around(ProceedingJoinPoint pjp){Object[] args = pjp.getArgs();System.out.println(Arrays.toString(args));args[0] = 666;Object ret = null;try{ret = pjp.proceed(args);}catch(Throwable throwable){t.printStackTrace();}return ret;}//其他的略
}

   在catch方法中就可以获取到异常,至于获取到异常以后该如何处理,这个就和你的业务需求有关了。

▶ 抛出异常后通知获取异常

@Component
@Aspect
public class MyAdvice {@Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")private void pt(){}@AfterThrowing(value = "pt()",throwing = "t")public void afterThrowing(Throwable t) {System.out.println("afterThrowing advice ..."+t);}//其他的略
}

如何让原始方法抛出异常,方式有很多,

@Repository
public class BookDaoImpl implements BookDao {public String findName(int id,String password) {System.out.println("id:"+id);if(true){throw new NullPointerException();}return "itcast";}
}

注意:

四、AOP事务管理

1、Spring事务简介

▶ 相关概念

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

▷ 数据层有事务我们可以理解,为什么业务层也需要处理事务呢?举个简单的例子:

 ● 转账业务会有两次数据层的调用,一次是加钱一次是减钱
 ● 把事务放在数据层,加钱和减钱就有两个事务
 ● 没办法保证加钱和减钱同时成功或者同时失败
 ● 这个时候就需要将事务放在业务层进行处理。

Spring为了管理事务,提供了一个平台事务管理器`PlatformTransactionManager`

commit是用来提交事务,rollback是用来回滚事务。

PlatformTransactionManager只是一个接口,Spring还为其提供了一个具体的实现:

 从名称上可以看出,我们只需要给它一个DataSource对象,它就可以帮你去在业务层管理事务。其内部采用的是JDBC的事务。所以说如果你持久层采用的是JDBC相关的技术,就可以采用这个事务管理器来管理你的事务。而Mybatis内部采用的就是JDBC的事务,所以后期我们Spring整合Mybatis就采用的这个DataSourceTransactionManager事务管理器。

▶ 知识点

@EnableTransactionManagement

@Transactional 

2、Spring事务角色 

▶ 未开启Spring事务之前

● AccountDao的outMoney因为是修改操作,会开启一个事务T1
● AccountDao的inMoney因为是修改操作,会开启一个事务T2
● AccountService的transfer没有事务,
  ○ 运行过程中如果没有抛出异常,则T1和T2都正常提交,数据正确
  ○ 如果在两个方法中间抛出异常,T1因为执行成功提交事务,T2因为抛异常不会被执行
  ○ 就会导致数据出现错误

▶ 开启Spring的事务管理后

 ● transfer上添加了@Transactional注解,在该方法上就会有一个事务T
 ● AccountDao的outMoney方法的事务T1加入到transfer的事务T中
 ● AccountDao的inMoney方法的事务T2加入到transfer的事务T中
 ● 这样就保证他们在同一个事务中,当业务层中出现异常,整个事务就会回滚,保证数据的准确性。

▷ 概念:

 ● 事务管理员:发起事务方,在Spring中通常指代业务层开启事务的方法
 ● 事务协调员:加入事务方,在Spring中通常指代数据层方法,也可以是业务层方法

 注意 : 目前的事务管理是基于`DataSourceTransactionManager`和`SqlSessionFactoryBean`使用的是同一个数据源。

3、事务属性

▶ 事务配置

上面这些属性都可以在`@Transactional`注解的参数上进行设置。

● readOnly:true只读事务,false读写事务,增删改要设为false,查询设为true。

● timeout:设置超时时间单位秒,在多长时间之内事务没有提交成功就自动回滚,-1表示不设置超时时间。

● rollbackFor:当出现指定异常进行事务回滚

● noRollbackFor:当出现指定异常不进行事务回滚

  ○ noRollbackFor是设定对于指定的异常不回滚,这个好理解

  ○ rollbackFor是指定回滚异常,对于异常事务不应该都回滚么,为什么还要指定?

    这块需要更正一个知识点,并不是所有的异常都会回滚事务,比如下面的代码就不会回滚

public interface AccountService {/*** 转账操作* @param out 传出方* @param in 转入方* @param money 金额*///配置当前接口方法具有事务public void transfer(String out,String in ,Double money) throws IOException;
}@Service
public class AccountServiceImpl implements AccountService {@Autowiredprivate AccountDao accountDao;@Transactionalpublic void transfer(String out,String in ,Double money) throws IOException{accountDao.outMoney(out,money);//int i = 1/0; //这个异常事务会回滚if(true){throw new IOException(); //这个异常事务就不会回滚}accountDao.inMoney(in,money);}}

○ 出现这个问题的原因是,Spring的事务只会对`Error异常`和`RuntimeException异常`及其子类进行事务回顾,其他的异常类型是不会回滚的,对应IOException不符合上述条件所以不回滚,此时就可以使用rollbackFor属性来设置出现IOException异常不回滚

@Service
public class AccountServiceImpl implements AccountService {@Autowiredprivate AccountDao accountDao;@Transactional(rollbackFor = {IOException.class})public void transfer(String out,String in ,Double money) throws IOException{accountDao.outMoney(out,money);//int i = 1/0; //这个异常事务会回滚if(true){throw new IOException(); //这个异常事务就不会回滚}accountDao.inMoney(in,money);}}

● rollbackForClassName等同于rollbackFor,只不过属性为异常的类全名字符串

● noRollbackForClassName等同于noRollbackFor,只不过属性为异常的类全名字符串

● isolation设置事务的隔离级别

  ○ DEFAULT   :默认隔离级别, 会采用数据库的隔离级别
  ○ READ_UNCOMMITTED : 读未提交
  ○ READ_COMMITTED : 读已提交
  ○ REPEATABLE_READ : 重复读取
  ○ SERIALIZABLE: 串行化

▶ 事务传播行为

对于上述案例的分析:

 ● log方法、inMoney方法和outMoney方法都属于增删改,分别有事务T1,T2,T3
 ● transfer因为加了@Transactional注解,也开启了事务T
 ● 前面我们讲过Spring事务会把T1,T2,T3都加入到事务T中
 ● 所以当转账失败后,所有的事务都回滚,导致日志没有记录下来
 ● 这和我们的需求不符,这个时候我们就想能不能让log方法单独是一个事务呢?

要想解决这个问题,就需要用到事务传播行为,所谓的事务传播行为指的是:

 ● 事务传播行为:事务协调员对事务管理员所携带事务的处理态度。

具体如何解决,就需要用到之前我们没有说的`propagation属性`。

▷ 1.修改logService改变事务的传播行为

@Service
public class LogServiceImpl implements LogService {@Autowiredprivate LogDao logDao;//propagation设置事务属性:传播行为设置为当前操作需要新事务@Transactional(propagation = Propagation.REQUIRES_NEW)public void log(String out,String in,Double money ) {logDao.log("转账操作由"+out+"到"+in+",金额:"+money);}
}

运行后,就能实现我们想要的结果,不管转账是否成功,都会记录日志。

▶ 2.事务传播行为的可选值

   对于我们开发实际中使用的话,因为默认值需要事务是常态的。根据开发过程选择其他的就可以了,例如案例中需要新事务就需要手工配置。其实入账和出账操作上也有事务,采用的就是默认值。

相关文章:

AOP面向切面编程思想。

目录 一、AOP工作流程 1、基本概念 2、AOP工作流程 二、AOP核心配置 1、AOP切入点表达式 2、AOP通知类型 三、AOP通知获取数据 1、获取参数 2、获取返回值 3、获取异常 四、AOP事务管理 1、Spring事务简介 2、Spring事务角色 3、事务属性 一、AOP工作流程 1、…...

实验7-变治技术及动态规划初步

目录 1.统计个数 2.数塔dp -A 3.Horspool算法 4.计数排序 5.找零问题1-最少硬币 1.统计个数 【问题描述】有n个数、每个元素取值在1到9之间,试统计每个数的个数 【输入形式】第一行,n的值;第二行࿰...

JVM垃圾回收机制GC理解

目录JVM垃圾回收分代收集如何识别垃圾引用计数法可达性分析法引用关系四种类型&#xff1a; 强、软、弱、虚强引用软引用 SoftReference弱引用 WeakReferenceWeakHashMap软引用与虚引用的使用场景虚引用与引用队列引用队列虚引用 PhantomReference垃圾回收算法引用计数复制 Cop…...

C++中的容器

1.1 线性容器1&#xff09;std::array看到这个容器的时候肯定会出现这样的问题&#xff1a;为什么要引入 std::array 而不是直接使用 std::vector&#xff1f;已经有了传统数组&#xff0c;为什么要用 std::array?先回答第一个问题&#xff0c;与 std::vector 不同&#xff0c…...

2023备战金三银四,Python自动化软件测试面试宝典合集(五)

接上篇八、抓包与网络协议8.1 抓包工具怎么用 我原来的公司对于抓包这块&#xff0c;在 App 的测试用得比较多。我们会使用 fiddler 抓取数据检查结果&#xff0c;定位问题&#xff0c;测试安全&#xff0c;制造弱网环境;如&#xff1a;抓取数据通过查看请求数据&#xff0c;请…...

SpringDI自动装配BeanSpring注解配置和Java配置类

依赖注入 上篇博客已经提到了DI注入方式的构造器注入&#xff0c;下面采用set方式进行注入 基于set方法注入 public class User {private String name;private Address address;private String[] books;private List<String> hobbys;private Map<String,String>…...

2月面经:真可惜...拿了小米的offer,字节却惨挂在三面

我是2月份参加字节跳动和华为的面试的&#xff0c;虽然我只拿下了小米的offer&#xff0c;但是我自己也满足了&#xff0c;想把经验分享出来&#xff0c;进而帮助更多跟我一样想进大厂的同行朋友们&#xff0c;希望大家可以拿到理想offer。 自我介绍 我是16年从南京工业大学毕…...

磐云PY-B8 网页注入

文章目录1.使用渗透机场景windows7中火狐浏览器访问服务器场景中的get.php&#xff0c;根据页面回显获取Flag并提交&#xff1b;2.使用渗透机场景windows7中火狐浏览器访问服务器场景中的post.php&#xff0c;根据页面回显获取Flag并提交&#xff1b;3.使用渗透机场景windows7中…...

多传感器融合定位十-基于滤波的融合方法Ⅰ其二

多传感器融合定位十-基于滤波的融合方法Ⅰ其二3. 滤波器基本原理3.1 状态估计模型3.2 贝叶斯滤波3.3 卡尔曼滤波(KF)推导3.4 扩展卡尔曼滤波(EKF)推导3.5 迭代扩展卡尔曼滤波(IEKF)推导4. 基于滤波器的融合4.1 状态方程4.2 观测方程4.3 构建滤波器4.4 Kalman 滤波实际使用流程4…...

Java集合面试题:HashMap源码分析

文章目录一、HashMap源码二、HashMap数据结构模型图三、HashMap中如何确定元素位置四、关于equals与hashCode函数的重写五、阅读源码基本属性参考文章&#xff1a;史上最详细的 JDK 1.8 HashMap 源码解析参考文章&#xff1a;Hash详解参考文章&#xff1a;hashCode源码分析参考…...

华为OD机试 - 数组合并(Python),真题含思路

数组合并 题目 现在有多组整数数组, 需要将他们合并成一个新的数组。 合并规则, 从每个数组里按顺序取出固定长度的内容合并到新的数组中, 取完的内容会删除掉, 如果该行不足固定长度或者已经为空, 则直接取出剩余部分的内容放到新的数组中, 继续下一行。 如样例 1, 获得长度…...

Vue2创建移动端项目

一、Vscode Vscode 下载安装以及常用的插件 1、Vscode 下载 下载地址&#xff1a;Vscode 中文语言插件 搜索 chinese 主题 Atom 主题 文件图标主题 搜索 icon 源代码管理插件GitLens 搜索 GitLens Live Server _本地服务器 搜索 Live Server Prettier - Code formatt…...

PorterDuffXfermode与圆角图片

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl 圆角图片 在项目开发中&#xff0c;我们常用到这样的功能&#xff1a;显示圆角图片。 这个是咋做的呢&#xff1f;我们来瞅瞅其中一种实现方式 /*** param bitmap 原图* p…...

如何准备大学生电子设计竞赛

大学生电子设计竞赛难度中上&#xff0c;一般有好几个类型题目可以选择&#xff0c;参赛者可以根据自己团队的能力、优势去选择合适自己的题目&#xff0c;灵活自主空间较大。参赛的同学们可以在暑假好好学习相关内容&#xff0c;把往年的题目拿来练练手。这个比赛含金量还是有…...

【Java容器(jdk17)】ArrayList深入源码,就是这么简单

ArrayList深入源码一、ArrayList源码解析1. MIXIN 的混入2. 属性说明3. 构造方法4. 其他方法&#xff08;核心&#xff09;iterator 和 listIterator 方法add方法remove 方法sort方法其他二、ArrayList 为什么是线程不安全的&#xff1f;体现哪些方面呢&#xff1f;三、ArrayLi…...

【Java 面试合集】简述下Java的三个特性 以及项目中的应用

简述下Java的特征 以及项目中的应用 1. 概述 上述截图中就是Java的三大特性&#xff0c;以及特性的实现方案。接下来就每个点展开来说说 2. 封装 满足&#xff1a;隐藏实现细节&#xff0c;公开使用方法 的都可以理解为是封装 而实现封装的有利手段就是权限修饰符了。可以根据…...

git基本概念图示【学习】

基本概念工作区&#xff08;Working Directory&#xff09;就是你在电脑里能看到的目录&#xff0c;比如名字为 gafish.github.com 的文件夹就是一个工作区本地版本库&#xff08;Local Repository&#xff09;工作区有一个隐藏目录 .git&#xff0c;这个不算工作区&#xff0c…...

微前端qiankun架构 (基于vue2实现)使用教程

工具使用版本 node --> 16vue/cli --> 5 创建文件 创建文件夹qiankun-test。 使用vue脚手架创建主应用main和子应用dev 主应用 安装 qiankun: yarn add qiankun 或者 npm i qiankun -S 使用qiankun&#xff1a; 在 utils 内创建 微应用文件夹 microApp,在该文件夹…...

记录robosense RS-LIDAR-16使用过程3

一、wireshark抓包保存pcap文件并解析ubuntu18安装wireshark&#xff0c;参考下面csdn教程&#xff0c;官网教程我看的一脸蒙&#xff08;可能英语太差&#xff09;https://blog.csdn.net/weixin_46048542/article/details/121730448?spm1001.2101.3001.6650.2&utm_medium…...

【博学谷学习记录】大数据课程-学习第七周总结

Hadoop配置文件修改 Hadoop安装主要就是配置文件的修改&#xff0c;一般在主节点进行修改&#xff0c;完毕后scp下发给其他各个从节点机器 文件中设置的是Hadoop运行时需要的环境变量。JAVA_HOME是必须设置的&#xff0c;即使我们当前的系统中设置了JAVA_HOME&#xff0c;它也…...

154、【动态规划】leetcode ——494. 目标和:回溯法+动态规划(C++版本)

题目描述 原题链接&#xff1a;494. 目标和 解题思路 &#xff08;1&#xff09;回溯法 本题的特点是nums中每个元素只能使用一次&#xff0c;分别试探加上nums[index]和减去nums[index]&#xff0c;然后递归的遍历下一个元素index 1。 class Solution { public:int res …...

MySQL-窗口函数

窗口函数概念常用窗口函数聚合窗口函数专用窗口函数语法OVER子句window_specwindow_name (命名窗口)partition_clause 分区order_clause 排序frame_clause 范围 &#xff08;指定窗口大小&#xff09;使用限制练习准备概念 窗口函数对一组查询执行类似于聚合的操作。然而&#…...

【C++设计模式】学习笔记(1):面向对象设计原则

目录 简介面向对象设计原则(1)依赖倒置原则(DIP)(2)开放封闭原则(OCP)(3)单一职责原则(SRP)(4)Liskov替换原则(LSP)(5)接口隔离原则(ISP)(6)优先使用对象组合,而不是类继承(7)封装变化点(8)针对接口编程,而不是针对实现编程结语简介 Hello! 非常感谢您阅读海…...

[测开篇]设计测试用例的方法如何正确描述Bug

​ 文章目录为什么测试人员要写测试用例&#xff1f;怎样设计测试用例&#xff1f;&#xff08;总的方面&#xff09;1.基于需求设计测试用例&#xff08;总的方面&#xff09; 2.页面&#xff08;总的方面&#xff09; 3.非功能性测试&#xff08;具体方面&#xff09; 4.1 等…...

设计模式学习笔记--单例、建造者、适配器、装饰、外观、组合

以下内容根据以下网址及相关视频整理&#xff1a;Android设计模式之单例模式_谬谬清不给我取名字的博客-CSDN博客_android 单例模式 Android设计模式--单例模式的六种实现和单例模式讲解Volatile与Synchronized相关的并发_龙腾腾的博客-CSDN博客_android 单例 volatile java …...

English Learning - Day5 L1考前复习 2023.2.10 周五

English Learning - Day5 L1考前复习 2023.2.10 周五1 单选题&#xff1a;She has the face _________.2 单选题&#xff1a; The goals ________ he fought all his life no longer seemed important to him.3 单选题&#xff1a;Sales director is a position ______ communi…...

C. Prepend and Append

time limit per test 1 second memory limit per test 256 megabytes input standard input output standard output Timur initially had a binary string†† s&#xfffd; (possibly of length 00). He performed the following operation several (possibly zero)…...

javassm超市在线配送管理系统

为了解决用户便捷地在网上购物&#xff0c;本文设计和开发了一个超市管理系统。本系统是基于web架构设计&#xff0c;SSM框架 &#xff0c;使用Mysql数据库管理&#xff0c;综合采用JSP模式来完成系统的相关功能。主要实现了管理员与用户的注册与登陆&#xff0c;个人中心、用户…...

Scratch少儿编程案例-多模式贪吃蛇(无尽和计时)

专栏分享 点击跳转=>Unity3D特效百例点击跳转=>案例项目实战源码点击跳转=>游戏脚本-辅助自动化点击跳转=>Android控件全解手册点击跳转=>Scratch编程案例👉关于作者...

谷歌蜘蛛池怎么搭建?Google蜘蛛池可以帮助谷歌排名吗?

本文主要分享关于谷歌蜘蛛池的搭建疑问&#xff0c;以及Google对谷歌排名的影响到底有多大。 本文由光算创作&#xff0c;有可能会被剽窃和修改&#xff0c;我们佛系对待这种行为吧。 谷歌蜘蛛池怎么搭建&#xff1f; 答案是&#xff1a;需要一个内链外链体系复杂的站群系统…...

公司设计网站应遵守哪些常理/站长工具seo综合查询广告

一、使用视图的理由是什么&#xff1f;1.安全性。一般是这样做的:创建一个视图&#xff0c;定义好该视图所操作的数据。之后将用户权限与视图绑定。这样的方式是使用到了一个特性&#xff1a;grant语句可以针对视图进行授予权限。2.查询性能提高。3.有灵活性的功能需求后&#…...

韩国化妆品网站金色flash片头/网站推广搜索

终于鼓起勇气想在图书馆预习&#xff08;。 &#xff09;下电路的&#xff0c;果然一上午又花在Java入门上了... Applet 其实就是一段Java代码&#xff0c;但这段代码可以以适当的方式嵌入到HTML页面。 1、编写一个Java Applet 用记事本编写如下代码&#xff0c;保存在某一目录…...

梧州网站建设定制/交友网站有哪些

翻过高山走不出你ob是output buffering的简称&#xff0c;就是输出缓冲区。如果使用了ob_start函数&#xff0c;那么之后的输出内容(echo等)就不进行实际输出&#xff0c;而是存入缓冲区里面&#xff0c;随后可以使用ob_flush实际输出、ob_clean删除、ob_get_contents获得内容保…...

网站免费观影怎么做/揭阳新站seo方案

系列文章目录 兼容Oracle与MySQL的那些事 兼容Oracle与MySQL的那些事&#xff08;分页问题&#xff09; 兼容Oracle与MySQL的一些事 兼容Oracle与MySQL的那些事【LISTAGG与GROUP_CONCAT】...

做外国的独立网站怎么推广/百度搜索关键词热度

2019独角兽企业重金招聘Python工程师标准>>> http://flywind.org/newtechnologydetail/175 转载于:https://my.oschina.net/flywind/blog/82647...

区块链做网站都有哪些内容呢/电商平台有哪些

尝鲜使用Kotlin写了一段时间Android。说大幅度的减少了Java代码一点不夸张。用Java的时候动不动就new一个OnClickListener()匿名类&#xff0c;动不动就类型转换的地方都可以省下很多。更不用说特殊的地方使用data class更是少些不知道多少代码。 Jetbrains给Android带来的不仅…...