0

晚上好,我在使用带有 JSF 的 Hibernate 时遇到了上述异常,我过去多次看到它,根本原因是这样,<session-factory name="sessionFactory">所以我删除了名称并更改了生成的代码以从中创建 SessionFactory:

protected SessionFactory getSessionFactory() {
    try {
        return (SessionFactory) new InitialContext()
                .lookup("SessionFactory");
    } catch (Exception e) {
        log.error("Could not locate SessionFactory in JNDI", e);
        throw new IllegalStateException(
                "Could not locate SessionFactory in JNDI");
    }
}

对此:

protected SessionFactory getSessionFactory() {
    try {
        return new Configuration().configure("hibernate.cfg.xml").buildSessionFactory();
    } catch (Exception e) {
        log.error("Could not locate SessionFactory in JNDI", e);
        throw new IllegalStateException(
                "Could not locate SessionFactory in JNDI");
    }
}

它对我来说工作得很好,但是这次我没有解决它,你知道问题出在哪里吗?

hibernate-cfg.xml

<hibernate-configuration>
<session-factory>
    <property name="hibernate.bytecode.use_reflection_optimizer">false</property>
    <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
    <property name="hibernate.connection.password">root</property>
    <property name="hibernate.connection.url">jdbc:mysql://localhost:3306/GUNO</property>
    <property name="hibernate.connection.username">root</property>
    <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
    <property name="hibernate.search.autoregister_listeners">false</property>
</session-factory>

4

2 回答 2

1

Sorry to necro this, but I thought that since this wasn't marked as answered perhaps it would be OK. Others may have fallen victim to the same pitfall as I.

I recently ran into similar problems after merging some HBM's. Subsequently, I had issues regarding JNDI and sessionFactory look-up failures due to duplicate class mappings in the HBM files causing the JNDI/SessionFactory service to be unable to start up..

So, as a start - make sure you are not declaring duplicate classes in your mapping files :)

This may or may not be what someone looking at this question needs, but it was my issue :)

于 2013-11-12T19:58:33.360 回答
0

您可以手动将 SessionFactory 添加到上下文中。虽然看起来很多代码,但实际上只有这 5 行代码。剩下的只是处理 InitialContext 似乎喜欢抛出的 NamingException。

更好的方法是使用 ContextListener 在启动期间自动添加会话

InitialContext initialContext = new InitialContext();
SessionFactory sf = (SessionFactory)  initialContext.lookup("SessionFactory");
Configuration cfg = new Configuration();
cfg.configure();
sf = cfg.buildSessionFactory();
initialContext.bind("SessionFactory", sf);

这是完整的 Servlet goGet 方法

    protected void doGet(HttpServletRequest request,
        HttpServletResponse response) throws ServletException, IOException {
    Account acc;
    InitialContext initialContext = null;
    acc = new Account("asdf" + String.valueOf(new Date().getTime()), "asdf");
    AccountHome home;
    Transaction tx = null;
    SessionFactory sf;

            // Create an instance of the InitialContext
            // So that we can lookup the SessionFactory property
            // or add it if it does not yet exist
    try {

        initialContext = new InitialContext();

    } catch (NamingException e) {
        throw new ServletException("Unable to create InitalContext", e);
    }

            // Since SessionFactories are very expensive to create 
            // first attempt to lookup a cached instance of the SessionFactory
    try {
        sf = (SessionFactory) initialContext.lookup("SessionFactory");
    } catch (NamingException e) {
                    // There is currently no session factory bound to this context
                    // Manually create it and bind it
        Configuration cfg;
        cfg = new Configuration();
        cfg.configure();
        sf = cfg.buildSessionFactory();

        try {
            initialContext.bind("SessionFactory", sf);
        } catch (NamingException e1) {
            throw new ServletException(
                    "Unable to bind the SessionFactory to the Inital Context");
        }
    }

            // Start the transaction and perform work
    tx = sf.getCurrentSession().beginTransaction();
    try {
        home = new AccountHome();
        home.persist(acc);
        tx.commit();
    } catch (Exception e) {
        tx.rollback();
        throw new ServletException("Work failed", e);
    }

}

编辑:添加了 ContextListener

package ch.yaawi.platform;

import javax.naming.InitialContext;
import javax.naming.NamingException;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;

import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

public class SessionFactoryListener implements ServletContextListener {

private SessionFactory mSessionFactory;

public void contextDestroyed(ServletContextEvent event) {

    if (mSessionFactory != null && !mSessionFactory.isClosed()) {
        mSessionFactory.close();
    }

}

public void contextInitialized(ServletContextEvent event) {
    InitialContext initialContext = null;
    try {

        initialContext = new InitialContext();

    } catch (NamingException e) {
        throw new RuntimeException("Unable to create InitalContext", e);
    }

    try {
        mSessionFactory = (SessionFactory) initialContext
                .lookup("SessionFactory");
    } catch (NamingException e) {
        Configuration cfg;
        cfg = new Configuration();
        cfg.configure();
        mSessionFactory = cfg.buildSessionFactory();

        try {
            initialContext.bind("SessionFactory", mSessionFactory);
        } catch (NamingException e1) {
            throw new RuntimeException(
                    "Unable to bind the SessionFactory to the Inital Context");
        }
    }

}

}

然后在你的 web.xml

<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
<display-name>Archetype Created Web Application</display-name>

<listener>
    <listener-class>ch.yaawi.platform.SessionFactoryListener</listener-class>
</listener>

</web-app>

导致将名称空间更改为您自己的名称空间。

希望这可以帮助某人

于 2013-05-08T08:04:14.170 回答