当前位置: 首页 > 知识库问答 >
问题:

spring data jpa saveandflush方法

令狐翰
2023-03-14

我确实理解JpaRepository Spring Data jpa类的方法save和方法saveAndFlush之间的区别。按照我的理解,save方法将只在事务结束时运行并提交sql,而saveAndFlush方法将通过运行sql语句而不提交它来使持久性上下文与数据库同步。下面是一个示例代码,我想在那里体验它,请回顾它。

这是更新的存储库类

@Repository
public interface ClassRepository extends JpaRepository<ClassA, Long> {

    @Modifying(clearAutomatically = true)
    @Query(value = "UPDATE class e SET e.class_name = ? WHERE e.employee_id = ?", nativeQuery = true)
    int updateClassNative(String className, String empId);

}

这是我测试方法的测试用例

    @Test
    void saveAndUpdateWithFlushJPA() {
        ClassA classA = ClassA.builder().className("Test").employeeId("S0810").build();

    this.classRepository.save(classA);
    int size = this.classRepository.updateClassNative("TestQ", "S0810");
    assertThat(size).isEqualTo(1);
}

阿什利

共有1个答案

傅鸿波
2023-03-14

您的测试场景的问题是,JPA总是在执行本机查询之前刷新持久性上下文(这也是JPQL查询的默认行为,尽管它可以被覆盖)。其基本原理是,查询应该报告反映当前工作单元中已经进行的更改的状态。

要查看save/saveandflush之间的区别,您可以使用以下测试用例来代替:

@Repository
public interface ClassRepository extends JpaRepository<ClassA, Long> {

    @Query("SELECT COUNT(c.id) FROM ClassA c")
    @QueryHints({
        @QueryHint(name = org.hibernate.annotations.QueryHints.FLUSH_MODE, value = "COMMIT")
    })
    int countClassAEntities();
}

@Test
@Transactional
void saveAndUpdate() {
    int initialCount = classRepository.countClassAEntities(); 
    ClassA classA = ClassA.builder().className("Test").employeeId("S0810").build();
    classRepository.save(classA);
    int finalCount = classRepository.countClassAEntities();
    assertEquals(initialCount, finalCount);
}

@Test
@Transactional
void saveAndUpdateWithFlush() {
    int initialCount = classRepository.countClassAEntities(); 
    ClassA classA = ClassA.builder().className("Test").employeeId("S0810").build();
    classRepository.saveAndFlush(classA);
    int finalCount = classRepository.countClassAEntities();
    assertEquals(initialCount + 1, finalCount);
}

在上面的设置中,count查询将刷新模式设置为commit,这意味着执行查询不会触发刷新。如果使用默认的repository.count()方法,则第一个测试用例将失败,因为默认情况下,刷新模式设置为auto

 类似资料:
  • 例如,我如何做到这一点? 和 像这样的东西,我试着在网上搜索,但我找不到任何关于这方面的教程,所以我在这里问。

  • 只是想知道为什么 调用world\u开关: 结果没有任何变化,我想增加或减少实例变量@pos\u X或@pos\u Y 这是我的初始化方法 这就是我如何创建类的实例 我们将不胜感激

  • 本文向大家介绍iOS hitTest方法&pointInside方法相关面试题,主要包含被问及iOS hitTest方法&pointInside方法时的应答技巧和注意事项,需要的朋友参考一下 hitTest方法 当事件传递给控件的时候,就会调用控件的这个方法,去寻找最合适的view point:当前的触摸点,point这个点的坐标系就是方法调用者   pointInside方法 作用:判断当前这个

  • 书籍与公开课 我喜欢看一些有关基础原理的书,如操作系统、计算机网络等。但是最近在看《计算机网络:自顶向下方法》的时候发现,这些书都太大而全了,让人看着很吃力,而且抓不住重点。 知名大学都有公开课,视频比枯燥的书要更好,也讲的更有侧重点。我觉得以后可以跟着公开课来看书: 如果不知道某个科目要看什么书,可以跟着公开课学习,看参考教材 如果已经确定要读某本经典书,可以先找有没有公开课 这些公开课可以在知

  • 主要内容:Python类实例方法,Python类方法,Python类静态方法和类属性一样,类方法也可以进行更细致的划分,具体可分为 类方法、 实例方法和 静态方法。 和类属性的分类不同,对于初学者来说,区分这 3 种类方法是非常简单的,即采用 @classmethod 修饰的方法为类方法;采用 @staticmethod 修饰的方法为静态方法;不用任何修改的方法为实例方法。 其中 @classmethod 和 @staticmethod 都是函数装饰器,后续章节会对其做详

  • Methods 方法 Pointers vs. Values 指针 vs. 值 As we saw with ByteSize, methods can be defined for any named type (except a pointer or an interface); the receiver does not have to be a struct. 正如 ByteSize 那样

  • 方法 是关联了特定类型的函数。类,结构体以及枚举都能定义实例方法,方法封装了给定类型特定的任务和功能。类,结构体和枚举同样可以定义类型方法,这是与类型本身关联的方法。类型方法与 Objective-C 中的类方法相似。 事实上在 结构体和枚举中定义方法是 Swift 语言与 C 语言和 Objective-C 的主要区别。在 Objective-C 中,类是唯一能定义方法的类型。但是在 Swift

  • 本页包含内容: 实例方法(Instance Methods) 类型方法(Type Methods) 方法是与某些特定类型相关联的函数。类、结构体、枚举都可以定义实例方法;实例方法为给定类型的实例封装了具体的任务与功能。类、结构体、枚举也可以定义类型方法;类型方法与类型本身相关联。类型方法与 Objective-C 中的类方法(class methods)相似。 结构体和枚举能够定义方法是 Swif