3

我将休眠实体存储在 ehcache 中。当调用外观层以检索实体时,我的拦截器将调用该方法并将其缓存。下次调用相同的方法时,实体将从缓存中返回。这一切都很好。

我的实体有一些定义为 FetchType.Lazy 的属性(对象或关联实体)。它是这样的,

@JoinColumn(name = "inventory_item_oid", referencedColumnName = "inventory_item_oid")
@ManyToOne(fetch = FetchType.LAZY)
private InventoryItem inventoryItem;

因此,并非所有属性都已加载。当需要库存项目时,它会被调用。此调用引发 LazyInitialization 异常。

由于我的缓存值只存在一天,因此可以在过期之前多次调用它。

这些调用之一引发上述异常。

我发现使用长休眠会话,我可以解决这个问题。但它不起作用,因为我的是基于请求/响应的应用程序。

还有另一种方法,我需要在访问其属性之前检查 InventoryItem 是否为空,如果它为空,那么我需要单独获取该值并将其附加到父项。这似乎很好......但需要做很多工作,因为我有很多实体。

我想知道是否有任何其他方法可以获取那些定义为惰性的对象。

4

1 回答 1

1

您不应该自己在 EHCache 中缓存实体。相反,您应该将 Hibernate 配置为使用二级缓存,并使用 EHCache 作为其实现。

最终结果将是相同的:您的实体将被缓存,并且您将往返保存到数据库。但是通过使用二级缓存,一切都将是透明的:您将从会话中加载实体,而 Hibernate 会自动将它们存储在缓存中。如果您从会话中重新加载它们,Hibernate 将从缓存中获取它们。如果您更新实体,它将从缓存中逐出以确保下次重新加载新副本。

与您的手工解决方案的另一个巨大区别是 Hibernate 将返回您附加的实体。因此,如果您只是调用 cachedEntity.getNonCachedEntity(),Hibernate 将延迟加载非缓存实体,就像您根本没有缓存一样。

实体是来自数据库还是来自二级缓存都不会改变任何东西:如果一个属性是延迟加载的,并且如果在会话关闭后访问该属性,那么延迟属性必须在会话之前初始化已经关了。

调用Hibernate.initialize(foo.getInventotyItem())初始化它。

更多信息在Hibernate 文档中。

于 2012-06-06T21:14:28.033 回答