SpringDataJpa对拿到的对象进行set,但是不save,数据库也能自动更新,由于使用了注解 @Transactional事务进行处理

发布时间 2023-08-01 19:55:15作者: sunny123456

SpringDataJpa对拿到的对象进行set,但是不save,数据库也能自动更新,由于使用了注解 @Transactional事务进行处理
原文链接:https://blog.csdn.net/qq_19903753/article/details/103367252

SpringDataJpa对拿到的对象进行set,但是不save,数据库也能自动更新

概述

今天在进行code review的时候,发现有一段代码很奇怪,就是标题所说的,将数据库的实体拿出来后,执行setXxx()更新,但是最后没有对实体进行save操作,数据库居然也能自动更新了。

代码如下:

/**
 * 这是一个举例子的方法,从数据库拿到实体->进行set更新且不save
*/
@Transactional
public void exampleMethod() {
    Optional<Food> foodOptional = foodDAO.findById(1L);
    // 如果不存在
    if(!foodOptional.isPresent()) {
        logger.info("the food 1L is not found!");
        return;
    }
	// 拿到实体
    Food food = foodOptional.get();
    // 对实体进行set更新状态
    food.setState(FoodState.EATTED);
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

执行结果:

没想到,数据库里的 food :1L 的状态变成了EATTED,但是我们没有进行save操作呀。

分析

我们使用的foodDAO.findById(1L)方法,查出来的是数据库的持久化对象,SpringDataJpa在事务完成的时候,会自动提交修改,这是因为SpringDataJpa底层还是利用hibernate实现的,对于hibernate,我们需要去了解实体的状态。

这里去搜到了一篇文章:hibernate实体状态

Hibernate实体状态的定义
1.瞬态:
一个实体通过new操作符创建后,没有和Hibernate的Session建立关系,也没有手动赋值过该实体的持久化
标识(持久化标识可以认为是映射表的主键)。此时该实体中任何属性的更新都不会反映到数据库表中。

2.持久化:
当一个实体和Hibernate的Session创建了关系,并获取了持久化标识,而且在Hibernate的Session生命周期内
存在。此时针对该实体任何属性的更改都会直接影响到数据库表中一条记录对应字段的更新,即与数据库表同步。

3.脱管:
当一个实体和Hibernate的Session创建了关系,并获取了持久化标识,而此时Hibernate的Session生命周期结
束,实体的持久化标识没有被改动过。针对该实体任何属性的修改都不会及时反映到数据库表中。

出现没有save确自动更新的情况,就是因为我们拿到的是持久化实体,在Service层自动提交事务的时候,会根据持久化对象自动更新数据库信息!