项目中使用的saveOrUpdate失效,查了很多资料,终于有点眉目了。
我在项目的web中有配置openSsionInView,导致saveOrUpdate失效,查看了网上的说法,其原因是配置了openSessionInVIew,其为了保证session在使用过程中是同一个,它会自动将FlushMode设置为Never,此时是只能进行读取操作,如果进行其他操作则会报出以下异常:Write operations are not allowed in read-only mode (FlushMode.NEVER/MANUAL),解决的方法有:
<filter>
<filter-name>OpenSessionInViewFilter</filter-name>
<filter-class>
org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
</filter-class>
<init-param>
<param-name>sessionFactoryBeanName</param-name>
<param-value>sessionFactory</param-value>
</init-param>
<init-param>
<param-name> flushMode </param-name>
<param-value>AUTO </param-value>
</init-param>
</filter>
回归到正题:不懂的是,在一个session过程中是自动将FlushMode设置为Never,但当session关闭之后会自动恢复为默认的FlushMode设置。此外,我要强调的是在这个项目中我使用到了spring的事务管理。在这些方面折腾了几天,才了解到spring在进行事务管理的时候会自动将flushMode设置为AUTO状态。此时,我们可以通过配置spring的事务来解决这个问题。
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="save*" read-only="false" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
<tx:method name="get*" read-only="true" propagation="REQUIRED"/>
<tx:method name="do*" propagation="REQUIRED"/>
<tx:method name="*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="myPointcut" expression="execution(* com.zxs.myproject.service.*.*(..))" />
<aop:advisor advice-ref="txAdvice" pointcut-ref="myPointcut"/>
</aop:config>
网上有用的信息:
无论connection.autocommit=true还是connection.autocommit=false 无论是声明式事务还是编程式事务都是有效的(即SuperA方法中的suba方法成功,subb方法失败,是会回滚掉suba的)
这个只是设置conn默认值而已.
(一)如果默认为true
1.保存对象是不需要开事务的,读写随意.
2.session.flush()是有意义的,因为session.flush()的操作是可以生效的.
(二)如果默认值为false
1.保存对象必须使用事务,不管是声明式事务还是编程式事务.当然也可以采用重置autocommit的方式
2.session.flush()是在事务块里面是没有意义的,因为数据操作是要统一提交的,但是这里我并没用经过复杂的测试,稍后有时间我再尝试几种情况.
3.如果不在事务块内修改持久化对象后执行session.flush(),数据库会锁表.
总结:
session的autocommit模式也是可以修改的,所以默认为false,在复杂的情况下需要手工控制flush时可以修改autocommit
疑惑:
查阅了spring的源码,在使用spring的声明式事务时,会修改session的FlushMode为AUTO,很不解,查阅JDBCTransaction类的commit方法
-
Java code
-
if ( ! transactionContext.isFlushModeNever() && callback ) { transactionContext.managedFlush(); // if an exception occurs during flush, user must call rollback() }
实际上当FlushMode为MANUAL或NEVER时,提交的时候会通过managedFlush()遍历所有的session的flush方法,不知道在事务块里面spring为什么要修改FlushMode为AUTO,当事务结束时又会改回原有的mode.
附上讲解spring事务的链接:http://www.ibm.com/developerworks/cn/java/j-lo-spring-ts1/