Write operations are not allowed in read-only mode (FlushMode.MANUAL):

来源:互联网 发布:java单例读取文件 编辑:程序博客网 时间:2024/05/20 09:07

转载自:http://blog.csdn.net/cpf929/article/details/18800147


错误信息如下:

ERROR 2014-01-26 19:38:38,504  [org.apache.struts2.dispatcher.Dispatcher] - <Exception occurred during processing request: Write operations are not allowed in read-only mode (FlushMode.MANUAL): Turn your Session into FlushMode.COMMIT/AUTO or remove 'readOnly' marker from transaction definition.>
org.springframework.dao.InvalidDataAccessApiUsageException: Write operations are not allowed in read-only mode (FlushMode.MANUAL): Turn your Session into FlushMode.COMMIT/AUTO or remove 'readOnly' marker from transaction definition.
at org.springframework.orm.hibernate3.HibernateTemplate.checkWriteOperationAllowed(HibernateTemplate.java:1175)
at org.springframework.orm.hibernate3.HibernateTemplate$12.doInHibernate(HibernateTemplate.java:685)
at org.springframework.orm.hibernate3.HibernateTemplate$12.doInHibernate(HibernateTemplate.java:1)
at org.springframework.orm.hibernate3.HibernateTemplate.doExecute(HibernateTemplate.java:406)
at org.springframework.orm.hibernate3.HibernateTemplate.executeWithNativeSession(HibernateTemplate.java:374)
at org.springframework.orm.hibernate3.HibernateTemplate.save(HibernateTemplate.java:683)

问题只读模式下(FlushMode.NEVER/MANUAL)写操作不被允许:把你的Session改成FlushMode.COMMIT/AUTO或者清除事务定义中的readOnly标记。

原因:在Spring的声明事务管理中,有如下配置:

[html] view plaincopy在CODE上查看代码片派生到我的代码片
  1. <tx:attributes>  
  2.              <!-- 指定哪些方法需要加入事务  -->   
  3.               
  4.              <tx:method name="save*" propagation="REQUIRED" />   
  5.   
  6.             <tx:method name="delete*" propagation="REQUIRED" />   
  7.   
  8.             <tx:method name="update*" propagation="REQUIRED" />   
  9.   
  10.             <!-- read-only="true":其余方法只读格式,加强其安全性 -->   
  11.   
  12.             <tx:method name="*" read-only="true" propagation="NOT_SUPPORTED" />   
  13.         </tx:attributes>  

而我保存数据的方法名写成了  register();所以会匹配到最后一个,只读模式。

网上关于这个问题的解释:

错误原因:
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(声明式的事务)保护的方法有写权限,没受保护的则没有。

解决方法:

web.xml配置里添加
<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>singleSession</param-name>
<param-value>true</param-value> 
</init-param>
<init-param>
<param-name> flushMode </param-name>
<param-value>AUTO </param-value> 
</init-param>
</filter>
// 。。

然后还要再事务方法那刷新flush(),否则不会立刻存储进数据库

  this.getHibernateTemplate().save(。。);
  getHibernateTemplate().flush() ;


还有一种方法就是直接不用这个过滤器.


<filter-mapping>
<filter-name>OpenSessionInViewFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

如果在交给spring 管理的情况下,在beans.xml 里的配置

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

<aop:config>
<aop:pointcut id="bussinessService"
expression="execution(* com.fan.service.base.*.*(..))" />
<aop:advisor pointcut-ref="bussinessService"
advice-ref="txAdvice" />
</aop:config>

<tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
<tx:method name="get*" read-only="false" propagation="NOT_SUPPORTED"/>
<tx:method name="find*" read-only="false" propagation="NOT_SUPPORTED"/>
<tx:method name="save*" propagation="REQUIRED"/> // 如果不把save update delete都配置上,
<tx:method name="update*" propagation="REQUIRED"/> //这些操作会无效
<tx:method name="delete*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>


0 0
原创粉丝点击