hibernate中的saveOrUpdate失效

 

 项目中使用的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/

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值