一尘不染

从多个线程修改休眠实体

hibernate

我在了解线程安全的详细信息时遇到了问题Hibernate。我知道这Hibernate Sessions本身并不是线程安全的,所以我不会从多个线程访问它们。但是,我找不到有关Hibernate实体的线程安全性的任何信息。我可以在多线程中修改它们,而又仍然将它们附加到用于加载它们的会话中吗?

我不会使用延迟加载(我知道这会导致并发问题)。实体将被正确同步,并且hibernate将通过同步的getter访问它们。

我设想的方案:

  • 使用hibernate会话从数据库加载实体A,
  • 随后,从加载实体的线程之外的多个线程修改实体A,
  • 实体A一直保持连接到会话并处于持久状态,
  • 刷新会话,以便将修改与数据库同步。
  • 实体A保持连接到会话,因此可以重复进行该循环,并进行进一步的修改和刷新。

阅读 216

收藏
2020-06-20

共1个答案

一尘不染

这取决于修改的性质。如果您通过在另一个线程中创建,持久化另一个实体并将其与之关联来修改一个实体,则它将无法工作,因为另一个实体实例将被视为在第一个线程中是分离的。

除了上面的用例之外,从理论上讲,这仅在您不使用字节码检测工具进行脏检查的情况下才有效。Hibernate只会在需要刷新对象时检查它们是否脏;基本上,它并不关心您如何修改对象。

但是,不建议这样做。

首先,它可能与Hibernate / JPA的未来版本不兼容(可能存在更多限制,阻止并发访问实体)。

其次,解决方法非常简单:只需为要同时修改的数据创建DTO,将其提交进行处理,并在处理完成后更新实体。这样,代码更加清晰,不会出现与Hibernate线程相关的意外投诉,您可以灵活地使用其他有用的功能,例如延迟加载。

2020-06-20