JAVA ·

JAVA 异常报错:Write operations are not allowed in read-only mode 解决方法

之前写了个定时任务功能,方法名:searchSipListSendEmail,主要是查询数据并且发送邮件的。一直运行着没什么问题,后来将方法改进了一下,其中涉及了save数据到数据库中(记录每个发邮件的收件人)。

等再执行方法时发现这方法异常报错了。

报错信息

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.
报错信息就是说,只读模式不允许保存任何数据。

问题原因

我方法名用了search(查询),在spring的配置的是read-only="true" propagation="REQUIRED",DAO采用了Spring容器的事务管理策略,所以你在做save功能的时候是不允许操作的;

专业点的问题解析(摘自网络):

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事物配置中

  • 1、read-only="true" propagation="REQUIRED" 改为false;
  • 2、修改方法名,不存在read-only="true"定义的;

问题总结

这是一个不应该犯的错误。因为我这个错误,我在开发调试的时候是没问题的,测试环境也没问题,在生产环境出的问题。

参与评论