java - 使用 hibernate 和 jersey JSP viewables 实现按请求 session + 延迟获取

标签 java hibernate jsp jersey lazy-loading

假设在应用程序中您有一个名为“用户”的实体。每个用户可以拥有许多使用延迟获取加载的文章。

@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/

相关文章:

Java EE 6 java.lang.NoClassDefFoundError glassfish

java - 尽管条件正确,但 if 条件永远不会执行

java - 如何使用 hibernate util 优化 dao?

java - gxt 中的分页网格

hibernate - Java Persistence API - Hibernate - 无法构建 EntityManagerFactory

java - 找不到适合 jdbc :ucanaccess://C:\Users\Asim Iqbal\Documents\PersonInfo. accdb 的驱动程序

java - jython多线程

java - 比较不同类型的对象列表的有效方法

java - 寻找独立的 JSP 编译器

java - 为什么我无法获取jsp的思想路径?