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

Grails-为什么需要withTransaction?

麹渊
2023-03-14

首先是一些背景知识:我正在使用Spring Security的修改版本来执行Active Directory身份验证,并检查存储在数据库中的可能访问权限。这意味着在普通的Groovy类中有一个调用来从数据库中加载信息:

if (Holders.config.loadRolesFromDatabase)
{
  Set<DomainClassRole> roles = DomainClassUser.findByUsername(username)?.roles
  if (roles)
    authorities.addAll(roles.collect({ new SimpleGrantedAuthority('ROLE_' + it.name) }))
}

这在Hibernate 4.3.6.1和Tomcat 7.0.54上效果很好,但是,在升级(升级到4.3.10.18和8.0.14.1)后,它现在在调用动态查找器方法时会产生“HibernateException:找不到当前线程的会话”异常。在做了一些研究之后,我决定将这段代码包装在一个withTransaction块中:

if (Holders.config.loadRolesFromDatabase)
{
  DomainClassUser.withTransaction({
    Set<DomainClassRole> roles = DomainClassUser.findByUsername(username)?.roles
    if (roles)
      authorities.addAll(roles.collect({ new SimpleGrantedAuthority('ROLE_' + it.name) }))
  })
}

这修复了错误,但是,我不确定为什么需要这样做。我目前对withTransaction的理解是,它用于创建在出现异常等情况下可以回滚的事务。然而,我不需要在此执行任何回滚(都是只读调用),为什么我仍然需要事务来执行此调用?

共有3个答案

祁晟
2023-03-14

您应该使用和newsession()。根据文档,它从史前Grails 1.2.0开始就可用。

简短回顾:

  • with Session()-使当前Hibernate会话可用(如果它存在,否则您会得到“未为当前线程找到会话”)
  • with NewSession()-在不启动事务的情况下创建新的Hibernate会话(我相信最适合您的解决方案)
  • with Transation()-启动并提交事务(不完全是简单只读操作的最佳选择)

(在Grails 3.1.13上测试)

谷出野
2023-03-14

这就是为什么即使在读取数据时也需要事务的原因。所有数据库语句都必须在物理事务中注册,这样就不会像你不使用它们一样。当您没有显式具有事务边界时,只需在自动提交模式下进行调整,因此每个语句都在不同的数据库事务中运行(这会增加开销)。更不用说对连接池解决方案造成的过度压力了。

杜骏祥
2023-03-14

有一个静态的 withSession 方法,看起来它是你在这里需要的,但事实并非如此;不幸的是,它只使当前的Hibernate会话可用,但是如果没有活动/当前会话,则不会创建一个。但是使用Transaction可以,因为在Grails中使用Hibernate时,Spring PlatformTransactionManager是一个HibernateTransactionManager,它确保在事务期间有一个活动会话,并在提交之前刷新并关闭它(除非有显式或自动(异常触发)回滚)。

因此,以这种方式与Transaction一起使用有点黑客行为,因为您依赖于副作用,但是无论如何您都要访问数据库,因此事务开销(实际上只是对连接的初始调用以将自动提交设置为false,隔离级别等,以及最后的no-op提交调用)是次要的,并且通常不是问题。我们真正需要的是介于 withSessionwithTransaction 之间的一些东西,它使会话可用,并在当前代码的持续时间内创建一个会话,但没有不必要的事务。

 类似资料:
  • 互联网是超文本标记语言(HTML)页面的集合,它们彼此链接以形成概念性信息网络。随着时间的推移,静态资源数量增加,图像等更丰富的项目开始成为Web结构的一部分。 高级服务器技术允许动态服务器页面 - 其内容基于查询生成的页面。 很快,需要拥有更多动态网页才能获得动态超文本标记语言(DHTML)。一切都归功于JavaScript。在接下来的几年中,我们看到了跨帧通信,试图避免页面重新加载,然后在帧内

  • 当前信息时代,哪里都是应用程序。这些应用程序们不仅仅是运行人们工作场所的工具 - 它们现在正在经营人们的生活。 对即时响应的需求,完美的行为和更多的功能是前所未有的。 而且,当然,人们期望应用程序在不同类型的设备上运行平稳,特别是在移动设备上。 应用程序执行的速度与它所做的一样重要。 NGINX的核心功能,例如其具有高性能HTTP和反向代理服务器的大规模可扩展事件驱动架构,访问和带宽控制以及与各种

  • 开发人员和运营工程师是两个不同的组织团队,如果发现这两个团队在错误的轨道上协作,则表明需要DevOps。以下是两个团队经常出现的一些问题: 在DevOps之前,开发和运营团队完全孤立。 测试和部署是在设计构建之后完成的独立活动。因此,他们比实际构建周期消耗更多时间。 在不使用DevOps的情况下,团队成员将大量时间花在测试,部署和设计上,而不是构建项目。 手动代码部署会导致生产中出现人为错误 编码

  • 问题内容: 根据PreparedStatement.setNull的Java文档,“注意:您必须指定参数的SQL类型”。该方法需要列的SQL类型的原因是什么? 我注意到传递java.sql.Types.VARCHAR还可用于非varchar列。是否存在不适合使用VARCHAR的方案(某些列类型或某些数据库提供程序)? 谢谢。 问题答案: 根据PreparedStatement.setNull的Ja

  • 问题内容: 我知道的是,全局变量和静态变量存储在段中,而未初始化的数据存储在段中。我不明白的是,为什么我们有专用于未初始化变量的段?如果未初始化的变量在运行时分配了值,那么该变量是否仅仍存在于段中? 在以下程序中, 在段中,并且在段中;那是对的吗?如果我的理解是错误的,请纠正我。 另外,请考虑以下程序, 问题答案: 原因是减小程序大小。想象一下,您的C程序在嵌入式系统上运行,其中代码和所有常量都保

  • 问题内容: 为什么需要放入GUI更新代码? 为什么Swing本身无法在内部对其进行处理?为什么调用者必须关心swing如何处理UI更新? 问题答案: 摆动对象不是线程安全的。顾名思义,允许在以后的某个时间执行任务;但更重要的是,该任务将在AWT事件分配线程上执行。使用时r,任务是异步执行的;还有,直到任务完成执行后才会返回。