sunny123456

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::
  1796 随笔 :: 22 文章 :: 24 评论 :: 226万 阅读
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

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层自动提交事务的时候,会根据持久化对象自动更新数据库信息!

posted on   sunny123456  阅读(469)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· .NET10 - 预览版1新功能体验(一)
历史上的今天:
2021-08-01 $(document).ready和window.onload区别
2021-08-01 为什么script标签一般放在body下面
2021-08-01 idea添加jar和移除jar包的三种方式
2021-08-01 HTML转义字符大全(转)
2021-08-01 C#中 以固定长度 取字符串中的数据
2021-08-01 Oracle的substr函数简单用法和 C#中一样
2021-08-01 ORA-22835:缓冲区对于CLOB到CHAR转换而言太小
点击右上角即可分享
微信分享提示