java - Singleton DAO 实例在 HTTP 事务之间保留旧的关闭 session

标签 java hibernate singleton dao

我正在尝试使用 Hibernate 实现“每个 http 请求一个 session ”模式,它适用于第一个请求:servlet 的 doGet() 方法打开 session ,获取一些内容,然后关闭 session session 。

但是当我刷新浏览器时,我的 DAO Singleton 实例(其构造函数从 SessionFactory 获取 session )被第二次调用,但仍然使用旧的 session 对象(单例构造函数不会再次调用)。然后我收到“ session 已关闭”错误。

我猜想单例实例必须在 HTTP 请求之间保存在缓存中,所以:如何才能再次调用 DAO 单例构造函数? (或者另一个优雅的解决方案来拥有新的 SessionFactory session 对象?)

非常感谢

servlet:

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    try {
        // Gets the session, eventually creates one
        Session s = HibernateUtil.currentSession();

        // Gets data from singleton DAO instance
        MySingletonDAO o = MySingletonDAO.getInstance();
        List<Stuff> stuff = o.getAllTheStuff();

        // send it to the view
        request.setAttribute("foo",stuff);
        RequestDispatcher dispatcher = getServletContext().getRequestDispatcher(vue);
        dispatcher.forward(request, response);

    }
             /*  error handling blah blah */
    finally {
        // closing the session
        HibernateUtil.closeSession();
    }

MySingletonDAO.java:

public class MySingletonDAO {
    // Usual singleton syntax
    private static MySingletonDAO INSTANCE = new MySingletonDAO();
    public static MySingletonDAO getInstance() { return INSTANCE;}

    private Session session;

    private MySingletonDAO() {
        session = HibernateUtil.currentSession();
        System.out.println("This constructor is called only on the first HTTP transaction");
    }

    public List<Stuff> getAllTheStuff() {
        try {
            session.beginTransaction();
            Query q = session.createQuery("FROM StuffDBTable");
            session.getTransaction().commit();
            return (List<Stuff>) q.list();
        }
    }
}

经典的线程安全 HibernateUtil.java :

public class HibernateUtil {

    private static final SessionFactory sessionFactory;
    public static final ThreadLocal session = new ThreadLocal();

    static {
        try {
            // Creates the SessionFactory
            sessionFactory = new Configuration().configure().buildSessionFactory();
        } catch (HibernateException he) {
            throw new RuntimeException("Conf problem : "+ he.getMessage(), he);
        }
    }


    public static Session currentSession() throws HibernateException {
        Session s = (Session) session.get();
        // Opens a new Session, if this Thread has none
        if (s == null || !s.isOpen() ) {
            s = sessionFactory.openSession();
            session.set(s);
        }
        return s;
    }

    public static void closeSession() throws HibernateException {
        Session s = (Session) session.get();
        session.set(null);
        if (s != null)
            s.close();
    }
}

最佳答案

你所要求的没有意义:如果在每个请求时调用单例的构造函数,那么它就不再是单例了。 session 确实在请求结束时关闭,但 DAO 保留对 session 的引用,而不是每次调用时从 util 类获取它。

您的 DAO 代码应该是

public class MySingletonDAO {
    private static MySingletonDAO INSTANCE = new MySingletonDAO();
    public static MySingletonDAO getInstance() { return INSTANCE;}

    private MySingletonDAO() {
    }

    public List<Stuff> getAllTheStuff() {
        Session session = HibernateUtil.currentSession();
        try {
            session.beginTransaction();
            Query q = session.createQuery("FROM StuffDBTable");
            session.getTransaction().commit();
            return (List<Stuff>) q.list();
        }
    }
}

也就是说,事务应该以声明方式处理,并且应该在服务层而不是 DAO 层处理:事务通常使用多个 DAO,DAO 返回的实体应保持托管状态,并且所有访问和修改都应保持托管状态这些实体应该在交易内进行。

我强烈建议使用 Java EE 容器或 Spring 来为您处理事务和 session 。您还应该使用标准 JPA API,而不是专有的 Hibernate API。

关于java - Singleton DAO 实例在 HTTP 事务之间保留旧的关闭 session ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42256797/

相关文章:

language-agnostic - 单例在编程中的用途

spring - 向 Spring 询问 Singleton Bean 的 Prototype 实例?

java - spring bean 配置中的微流 EmbeddedStorageManager

java - 使用 Stripes 框架拦截和更改请求 url

java - Hibernate 枚举映射

mysql - Spring中调试 "org.springframework.beans.factory.BeanCreationExcpetion"错误

c# - 有初始化的设计模式吗? C#

java - 具有两个不同扫描仪和两个不同 BufferedReader 的相同 TCP 套接字(客户端)

java - 如何使用 ActionListener 从按钮调用面板

java - 从父级 JPA 获取所有子级和子级