1

我正在使用 Spring Framework v3.2.4 和 JPA + Hibernate 4 实现 RESTful 服务。我以 JSON 格式返回资源(使用 Jackson Mapper),但现在我遇到了这个异常:

Could not write JSON: failed to lazily initialize a collection of role: it.teck.service.model.Canvas.params, could not initialize proxy - no Session (through reference chain: it.teck.service.model.Canvas["params"]);

Canvas我在和实体之间有“多对多”关系,Param当向服务请求画布时,我还需要序列化参数列表。

在我的课上,我有:

@Entity
public class Canvass {

    @ManyToMany
    @JoinTable(name = "canvas_params", joinColumns = { @JoinColumn(name = "id_canvas", referencedColumnName = "id_canvas") }, inverseJoinColumns = { @JoinColumn(name = "id_param", referencedColumnName = "id_param") })
        private List<Param> params;

    // ...
}

和:

@Entity
public class Param {

    @ManyToMany(mappedBy = "params")
    private List<Canvas> canvasList;

    // ...
}

在一些帖子和 SO 答案中,建议从序列化中排除ManyToMany字段,以打破序列化循环,但我需要序列化链接到我的画布实体的参数。所以我该怎么做?

4

5 回答 5

4

发生这种情况是因为 Jackson 试图访问由 Hibernate 在会话之外管理的 bean 的属性。因此该属性是延迟加载的,当您尝试在会话之外访问它时,休眠将无法从数据库中获取它。

你有三个选择:

  1. 你确实像 Ion 所说的那样,在实体上设置了 FetchType.EAGER。缺点是每次您以这种方式获取一个实体时,它都会获取与其链接的所有 Canvases 和 Params。你可能不希望这样,因为它可能会减慢你的应用程序。
  2. 当仍在 DAO 级别或服务级别的会话中时,您可以对其进行序列化。这是最干净最合适的方式
  3. 最后,如果(仍在会话中)您执行 params.size() 或 canvasList.size() 之类的操作,这将自动触发从 db 获取集合。这有点小技巧,但它可以工作,您不必修改 DAO 签名或实体的获取策略。

对于 Spring,还有另一种选择,包括使发生序列化的控制器方法@Transactional,这将保持会话打开!但这会使该方法具有事务性,因此请注意不必要的后果。

于 2013-11-08T10:47:51.697 回答
4

由于在序列化时 Hibernate 延迟加载,您最终会使用 Hibernate 代理集合。此时会话已经关闭或不在范围内。您可以编写一个在范围内具有 JPA 实体管理器的自定义转换器(查找OpenEntityManagerInViewFilter示例),或者如果性能不是问题(集合相当小)并且您总是需要完全填充的对象,您可以指定一个急切的获取策略映射为:

@ManyToMany(fetch = FetchType.EAGER) 
@JoinTable(name = "canvas_params", joinColumns = { @JoinColumn(name = "id_canvas", referencedColumnName = "id_canvas") }, inverseJoinColumns = { @JoinColumn(name = "id_param", referencedColumnName = "id_param") })
    private List<Param> params;
于 2013-11-08T10:39:04.363 回答
0

提供的其他答案确实涵盖了问题的原因。对于解决方案,我建议您研究诸如 Hexagonal Architecture 之类的东西。

本质上,您需要一个存储库层,您可以在其中获取要使用的数据,通过强迫自己从实体数据对象映射到域数据对象,您实际上将绕过这个问题。原因是你正在分离你的担忧。您将在一个地方加载您将使用的所有数据(不是您不会使用的额外数据),然后您在不同的地方执行任何逻辑。这将产生首先完成所有数据库内容并释放这些资源的效果,以便您可以在闲暇时执行您的逻辑。

一个很好的链接是:http ://alistair.cockburn.us/Hexagonal+architecture

即使您的实施并不严格,应用它所讨论的一些规则也会有很大帮助。

希望这可以帮助。

于 2015-03-16T09:19:21.763 回答
0

您的问题似乎与序列化循环无关,而是与在检索父对象的事务之外完成的序列化有关。

要解决这个问题,你能在这个 tx 中进行序列化吗?

您还有其他几个选项,如休眠“延迟初始化异常”线程中所述:未能延迟初始化角色集合

于 2013-11-08T10:41:02.167 回答
0

您可以尝试如下在运行时更改获取策略以克服此问题。

User user = (User) session.createCriteria(User.class)
    .setFetchMode("permissions", FetchMode.JOIN)
    .add( Restrictions.idEq(userId) )
    .uniqueResult();

“FetchMode”中有几个选项。

于 2016-01-20T14:20:20.613 回答