假设在应用程序中您有一个名为“用户”的实体。每个用户可以拥有许多使用延迟获取加载的文章。
@Entity
@Table(name = "Users")
public class User {
private Integer id;
private Set<Article> articles = new HashSet<Article>();
...
@OneToMany(fetch = FetchType.LAZY, mappedBy = "user", cascade = CascadeType.ALL)
public Set<Article> getArticles() {...}
}
在Web服务端,我想通过Id加载用户并将其传递给jersey Viewable JSP以返回和显示。我还想对每个请求使用一个 session 。
@GET
public Response getUser() {
Session session = getSessionFactory().openSession();
session.beginTransaction();
User user = (User) session.getNamedQuery("getUserById").setInteger("id", 1).uniqueResult();
Map<String, Object> dataMap = new HashMap<String, Object>();
dataMap.put("user", user);
Viewable v = new Viewable("/user", dataMap);
Response r = Response.ok(v).build();
session.getTransaction().commit();
session.close();
return r;
}
在 user.jsp 中,我想访问文章并将其打印出来。
<c:forEach var="a" items="${it.user.articles}">
<span>${a.title}</span>
</c:forEach>
运行此代码并尝试请求 JSP 最终会引发异常:
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: my.package.User.articles, no session or session was closed
发生这种情况是因为显然创建 Viewable、将其传递给 ResponseBuilder 并调用 build() 实际上并没有创建 JSP。在我的方法返回并且 session 已关闭之后,JSP 会在更远的地方创建。
一个明显的选择是将文章的获取类型设置为 eager,但这并不理想,因为在大多数情况下,当我使用 User 对象时,我不需要加载或显示文章。
有没有办法强制 jersey 在我的方法返回之前解释和构造 jsp?是否有某种 onResponse 或 postResponse 监听器我可以传递我的代码来关闭 session ?
一般情况下,我处理打开/关闭 hibernate session 的方式是否应该采取不同的做法?通过谷歌搜索,每个请求一个 session 似乎是大多数人提倡使用的方式,所以我想我应该尝试一下。
最佳答案
"open session in view"方法确实是解决这个问题的方法。
另一种方法是使用 EJB 或 Spring。然后他们将接管交易处理。当与 JPA EntityManager 一起使用时,EJB 会完全透明地完成此操作,Spring 为此提供了 org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter 过滤器。
如果您的容器支持 EJB,则只需创建一个 @Stateless
:
@Stateless
public class UserService {
@PersistenceContext
private EntityManager em;
public User find(Integer id) {
return em.createNamedQuery("getUserById", User.class)
.setParameter("id", id)
.getSingleResult();
}
}
并在您的网络服务中使用它,如下所示:
@EJB
private UserService userService;
@GET
public Response getUser() {
User user = userService.find(1);
Map<String, Object> dataMap = new HashMap<String, Object>();
dataMap.put("user", user);
Viewable v = new Viewable("/user", dataMap);
Response r = Response.ok(v).build();
return r;
}
无需再担心事务和延迟获取。
关于java - 使用 hibernate 和 jersey JSP viewables 实现按请求 session + 延迟获取,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10705697/