【问题标题】:Hibernate 4 - Calling DAO and Initializing sessionFactory beanHibernate 4 - 调用 DAO 并初始化 sessionFactory bean
【发布时间】:2012-11-27 12:40:11
【问题描述】:

我正在将我的旧程序更新到 Spring 3.2 和 Hibernate 4,并且在使用 sessionFactory 时遇到了一些困难(我之前使用的是 hibernateTemplate)。

  1. 我不认为我访问 DAO 的方式是最好的方式,但我不知道还有什么方法可以让它发挥作用。如果我简单地创建 DAO 对象 (CSSDAO d = new CSSDAOImpl();),则 sessionFactory 始终为空。如果我按照下面的方式拥有它,它就可以工作。调用 DAO 方法的正确方法是什么? (请忽略控制器的 MVC 部分,我知道这需要它自己的工作)

  2. 我将在 DAO 中的每个方法中打开一个新会话。我知道这是不正确的,因为我应该得到当前会话。但是每次我尝试获取当前会话时,它都会说一个不存在。会话如何第一次“初始化”?我认为它会基于 XML 配置注入它,但这似乎没有做任何事情。有什么想法吗?

hibernate-cfg.xml

    <?xml version="1.0" encoding="UTF-8"?>


<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">

    <bean id="myDataSource"
        class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="org.apache.derby.jdbc.EmbeddedDriver"></property>
        <property name="url" value="jdbc:derby:C:\Users\Steven\MyDB"></property>
    </bean>

    <bean id="sessionFactory"
        class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="packagesToScan" value="net.form" />
        <property name="dataSource" ref="myDataSource"></property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.DerbyTenSevenDialect
                </prop>
                <prop key="hibernate.show_sql">true</prop>
            </props>
        </property>
        <property name="annotatedClasses">
            <list>
                <value>net.form.StyleChooser</value>
            </list>
        </property>
    </bean>

    <bean id="transactionManager"
        class="org.springframework.orm.hibernate4.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory" />
    </bean>

    <tx:annotation-driven transaction-manager="transactionManager" />

    <bean id="CSSDAO" class="dao.CSSDAOImpl">
        <property name="sessionFactory" ref="sessionFactory" />
    </bean>

</beans>

DAO:

package dao;

import java.util.List;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;

import net.form.StyleChooser;

public class CSSDAOImpl implements CSSDAO {

    private SessionFactory sessionFactory;

    @Autowired
    public void setSessionFactory(SessionFactory sessionFactory) {
        this.sessionFactory = sessionFactory;
    }

    @Transactional
    public List selectAllCSS() {
        Session session = sessionFactory.openSession();
        Transaction tx = session.beginTransaction();
        List l = session.createCriteria(StyleChooser.class).list();
        session.flush();
        tx.commit();
        return l;
    }

    @Transactional
    public StyleChooser selectCSSById(Integer ID) {
        Session session = sessionFactory.openSession();
        Transaction tx = session.beginTransaction();
        StyleChooser sc = (StyleChooser) session.get(StyleChooser.class, ID);
        session.flush();
        tx.commit();        
        return sc;
    }

    @Transactional
    public Integer insertCSS(StyleChooser insertCSS) {
        Session session = sessionFactory.openSession();
        Transaction tx = session.beginTransaction();
        Integer id = (Integer) session.save(insertCSS);
        session.flush();
        tx.commit();
        return id;
    }

    @Transactional
    public void deleteCSS(Integer CSSId) {
        Session session = sessionFactory.openSession();
        Transaction tx = session.beginTransaction();
        StyleChooser sc = (StyleChooser) session.get(StyleChooser.class, CSSId);
        session.delete(sc);
        session.flush();
        tx.commit();
    }

    @Transactional
    public void updateCSS(StyleChooser cssWithNewValues) {
        Session session = sessionFactory.openSession();
        Transaction tx = session.beginTransaction();
        session.update(cssWithNewValues);
        session.flush();
        tx.commit();        
    }
}

访问 DAO...

package net.controllers;

import java.util.List;
import javax.servlet.http.HttpServletRequest;
import net.form.StyleChooser;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.FileSystemResource;
import org.springframework.core.io.Resource;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.multiaction.MultiActionController;
import dao.CSSDAOImpl;

@Controller
@RequestMapping("/stylechoosertable.html")
public class IndexController extends MultiActionController {

    Resource resource = new FileSystemResource(
            "C:/Users/Steven/Desktop/Programming/workspace/CSSGeneratorHibernate4/WebContent/WEB-INF/hibernate.cfg.xml");
    BeanFactory beanFactory = new XmlBeanFactory(resource);
    CSSDAOImpl dao = (CSSDAOImpl) beanFactory.getBean("CSSDAO");

    @RequestMapping(method = RequestMethod.GET)
    public ModelAndView showIndex(HttpServletRequest request) throws Exception {
        List<StyleChooser> styleChooser = dao.selectAllCSS();
        return new ModelAndView("stylechoosertable", "styleChooser", styleChooser);
    }
}

【问题讨论】:

    标签: java spring hibernate


    【解决方案1】:

    很少观察:

    1. 在您的检索方法中,不应使用事务,即它们应该是非事务性的。

    2. 在您的配置中添加&lt;tx:annotation-driven transaction-manager="transactionManager"/&gt; 以识别事务注释。

    3. 如果您使用@Transactional 注释,则不需要使用程序化事务。在@Transactional注解中添加传播属性为@Transactional(propagation=Propagation.REQUIRED),并将事务管理留给Hibernate。

    4. 第一次您需要打开会话,如果没有关闭,您下次可以使用相同的会话。要获取session,最好使用如下实用方法:

      private Session getSession(SessionFactory sessionFactory){
       Session session = null;
       try{
           session = sessionFactory.getCurrentSession();
       }catch(HibernateException hex){
           hex.printStackTrace();
       }
       if(session == null && !session.isClosed()){
           session = sessionFactory.openSession();
       }
      }
      

    这样,您将获得会话(如果可用)并打开,否则打开一个新会话。

    【讨论】:

    • 感谢您的回答。所以我需要从方法中删除事务行并通过事务注释让事务休眠,对吗?为什么我必须打开一个新会话?我认为配置会通过注入创建一个 sessionFactory 和会话?不是这样吗?最后,应该如何准确访问 DAO(问题 1)
    • @user1795832:1. 正确。只需确保配置正确且事务按预期工作即可。 2. 您只是在创建会话工厂。您需要使用会话工厂获取/创建会话。 3. DAO访问模式参考community.jboss.org/wiki/GenericDataAccessObjects
    • 好的,很好用。但是每次调用 getSession 时我都会遇到异常,因此 sessionFactory 中的当前会话在任何情况下都不存在。这是为什么?会话退出事务边界后是否会被销毁?此外,对于 deleteCSS 和 updateCSS 方法,我还必须调用 flush() 以使查询通过,但我不必为 insertCSS 调用。有什么想法吗? PS抱歉,当我出于某种原因将@YOUR USER NAME 放在那里时,它会一直删除它,所以不确定您是否收到通知。
    • @user1795832 我怀疑你的 be 本身正在被重新创建,因此没有得到旧的会话。尝试设置 dao bean 的 scope。关于flush,如果您的交易正常进行,我认为这不是必需的。
    • 除非我刷新,否则不会进行更新和删除。我读过 Transactional 应该处理这个问题,但它似乎不起作用。
    猜你喜欢
    • 2013-09-25
    • 1970-01-01
    • 2012-06-10
    • 2013-06-07
    • 2011-05-06
    • 1970-01-01
    • 2017-05-04
    • 2013-06-11
    • 1970-01-01
    相关资源
    最近更新 更多