Write operations are not allowed in read-only mode问题处理

编程技术  /  houtizong 发布于 3年前   52

参考自:

http://hi.baidu.com/woiwojia/blog/item/08755f032f75607e3812bb8c.html

http://blog.sina.com.cn/s/blog_5dc12c490100crr5.html

 

错误代码:
org.springframework.dao.InvalidDataAccessApiUsageException: Write operations are not allowed in read-only mode (FlushMode.NEVER) - turn your Session into FlushMode.AUTO or remove 'readOnly' marker from transaction definition
错误原因:
这是因为使用Spring提供的Open Session In View而引起的问题,OpenSessionInViewFilter在getSession的时候,会把获取回来的session的flush mode 设为FlushMode.NEVER。然后把该sessionFactory绑定到TransactionSynchronizationManager,使request的整个过程都使用同一个session,在请求过后再接除该sessionFactory的绑定,最后closeSessionIfNecessary根据该session是否已和transaction绑定来决定是否关闭session。在这个过程中,若HibernateTemplate 发现自当前session有不是readOnly的transaction,就会获取到FlushMode.AUTO Session,使方法拥有写权限。
也即是,如果有不是readOnly的transaction就可以由Flush.NEVER转为Flush.AUTO,拥有insert,update,delete操作权限,如果没有transaction,并且没有另外人为地设flush model的话,则doFilter的整个过程都是Flush.NEVER。所以受transaction保护的方法有写权限,没受保护的则没有
解决办法:
采用spring的事务声明,使所有service方法受transaction控制

<!-- Spring2.0的事务管理 -->
 <bean id="transactionManager"
  class="org.springframework.orm.hibernate3.HibernateTransactionManager">
  <property name="sessionFactory">
   <ref local="sessionFactory" />
  </property>
 </bean>
 <tx:advice id="txtAdvice"
  transaction-manager="transactionManager" >
  <tx:attributes>
   <tx:method name="add*" propagation="REQUIRED" />
   <tx:method name="save*" propagation="REQUIRED" />
   <tx:method name="update*" propagation="REQUIRED" />
   <tx:method name="delete*" propagation="REQUIRED" />
   <tx:method name="find*" propagation="REQUIRED"
    read-only="true" />
   <tx:method name="get*" propagation="REQUIRED"
    read-only="true" />
   <tx:method name="search*" propagation="REQUIRED"
    read-only="true" />
   <tx:method name="*" propagation="REQUIRED" />
  </tx:attributes>
 </tx:advice>

 <aop:config proxy-target-class="true">
  <aop:pointcut id="serviceMethods"
   expression="execution(* com.service.*.*(..))" />
  <aop:pointcut id="baseServiceMethods"
   expression="execution(* com.base.service.*.*(..))" />
  <aop:advisor advice-ref="txtAdvice"
   pointcut-ref="serviceMethods" />
  <aop:advisor advice-ref="txtAdvice"
   pointcut-ref="baseServiceMethods" />

  <!--使不同包下的service方法都受事务管理-->
 </aop:config>

请勿发布不友善或者负能量的内容。与人为善,比聪明更重要!

留言需要登陆哦

技术博客集 - 网站简介:
前后端技术:
后端基于Hyperf2.1框架开发,前端使用Bootstrap可视化布局系统生成

网站主要作用:
1.编程技术分享及讨论交流,内置聊天系统;
2.测试交流框架问题,比如:Hyperf、Laravel、TP、beego;
3.本站数据是基于大数据采集等爬虫技术为基础助力分享知识,如有侵权请发邮件到站长邮箱,站长会尽快处理;
4.站长邮箱:[email protected];

      订阅博客周刊 去订阅

文章归档

文章标签

友情链接

Auther ·HouTiZong
侯体宗的博客
© 2020 zongscan.com
版权所有ICP证 : 粤ICP备20027696号
PHP交流群 也可以扫右边的二维码
侯体宗的博客