事务没有启动Spring + Hibernate + MySQL

时间:2014-02-06 11:02:52

标签: java mysql spring hibernate

我在Spring中的事务没有启动时出现问题,因此不会保存对数据库所做的更改。配置如下:

web.xml(没有dispatcherServlet,有FacesServlet)

<context-param>
    <param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

applicationContext.xml中的相关配置如下:

<context:annotation-config />
<context:component-scan base-package="pl.web.beans,pl.csci.dao" />

<bean id="sessionFactoryCsci" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
    <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />
    <property name="configLocation" value="classpath:csci_database.cfg.xml" />
</bean>

<bean id="hibernateTemplateCsci" class="org.springframework.orm.hibernate3.HibernateTemplate">
    <property name="sessionFactory">
        <ref bean="sessionFactoryCsci" />
    </property>
</bean>

<bean id="hibernateDaoSupportCsci" class="org.springframework.orm.hibernate3.support.HibernateDaoSupport" abstract="true">
    <property name="hibernateTemplate" ref="hibernateTemplateCsci" />
</bean>

<bean id="csciPermSystemDAO" class="pl.csci.dao.CsciPermSystemDAO" parent="hibernateDaoSupportCsci">
</bean>

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

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

最后负责保存实体的班级:

pl.csci.dao
@Transactional(readOnly = true)
public class CsciPermSystemDAO extends HibernateDaoSupport implements ICsciPermSystemDAO {
@Override
@Transactional(readOnly = false, propagation = Propagation.REQUIRES_NEW)
public Integer addPermission(Permissions Permission) throws DaoException {
    try {
       Integer permissionId = (Integer) getHibernateTemplate().save(Permission);
        LOGGER.info("Is transaction active?:" + DebugUtils.transactionActive() );
        return permissionId;
    } catch (Throwable ex) {
        // exception handling
    }
}

上面的代码片段(DebugUtils.transactionActive())返回false。此方法使用TransactionSynchronizationManager类来确定是否激活了事务。

调试Hibernate时,我可以看到来自插入的日志,但这些插入实际上从未出现在数据库中:

2014-01-30 15:13:15 DEBUG [hibernate.SQL] - 插入csci_perm.permissions(BRANCH,GROUP_ID,PERSON_ID,READ_PERM,SYSTEM_ID,WRITE_PERM)值(?,?,?,?,?, ?)

请告诉我配置有什么问题?

最后,这是hibernate配置:

<hibernate-configuration>
  <session-factory>
    <!-- Database connection settings -->
    <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
    <property name="hibernate.connection.url">jdbc:mysql://localhost:3306/csci_perm</property>
    <property name="hibernate.connection.useUnicode">true</property>
    <property name="hibernate.connection.characterEncoding">utf8</property>
    <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
    <property name="hibernate.connection.username">user</property>
    <property name="hibernate.connection.password">password</property>
    <property name="connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
    <property name="c3p0.acquire_increment">1</property>
    <property name="c3p0.idle_test_period">3000</property>
    <!-- seconds -->
    <property name="c3p0.max_size">20</property>
    <property name="c3p0.max_statements">50</property>
    <property name="c3p0.min_size">3</property>
    <property name="c3p0.timeout">100</property>
    <!-- seconds -->
    <!-- Enable Hibernate's automatic session context management -->
    <property name="current_session_context_class">thread</property>
    <!-- Disable the second-level cache  -->
    <property name="cache.provider_class">org.hibernate.cache.NoCacheProvider</property>
    <!-- Echo all executed SQL to stdout - You can disable this once you have it working -->
    <property name="show_sql">true</property>
    <property name="hibernate.jdbc.batch_size">10</property>
    <mapping class="pl.csci.model.Permissions"/>
  </session-factory>
</hibernate-configuration>

2 个答案:

答案 0 :(得分:0)

可能是数据库不支持事务。

选中document

  

MySQL Server(版本3.23-max和所有版本4.0及更高版本)支持与InnoDB和BDB事务存储引擎的事务。

答案 1 :(得分:0)

您的配置错误,使用spring时永远不会弄乱current_session_context_class,除非您使用的是JTA。将其设置为线程禁用弹簧正确的事务管理。

接下来你的代码也在破坏正确的事务管理永远不会捕获并吞下一个预期,这样transactionamangement失败,因为它永远不会看到异常并将尝试提交。捕获和重新抛出或不捕获。

最后,您不应该使用从Hibernate 3.0.1(2006年左右)起应被视为弃用的HibenateDaoSupport和/或HibernateTemplate。直接使用SessionFactory解释here

基本上你的dao应该是这样的

@Transactional(readOnly = true)
public class CsciPermSystemDAO implements ICsciPermSystemDAO {

    private SessionFactory sf;

    @Override
    @Transactional(readOnly = false, propagation = Propagation.REQUIRES_NEW)
    public Integer addPermission(Permissions Permission) throws DaoException {
        LOGGER.info("Is transaction active?:" + DebugUtils.transactionActive() );
        return (Integer) sf.getCurrentSession().save(permission);
    }

    public void setSessionFactory(SessionFactory sf) {
        this.sf=sf;
    }
}

另一个暗示hibernate.connection文件中的c3p0hibernate.cfg.xml属性无用。您正在注入数据源,这些属性仅在hibernate管理数据源/连接时使用。

关于您的弹簧配置<context:annotation-config />已隐含<context:component-scan />。在您的会话出厂配置中,您要设置org.hibernate.cfg.AnnotationConfiguration删除此内容,而是使用AnnotationSessionFactoryBean代替LocalSessionFactoryBean。这可以节省一些xml。