异常:Write operations are not allowed in read-only mode (FlushMode.NEVER)

本文探讨了在使用Spring框架时遇到的Writeoperationsarenotallowedinread-onlymode异常,通过调整事务配置和设置FlushMode.AUTO的方式成功解决了问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我用Spring的声明事务,增加一个业务功能<关闭问题>时,我出现了异常:
Write operations are not allowed in read-only mode (FlushMode.NEVER) - turn your Session into FlushMode.AUTO or remove 'readOnly' marker from transaction definition

查了错误原理:在业务层dao.update(quesModel)中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保护的方法有写权限,没受保护的则没有。
我的解决方式是:
1.将singleSession设为false,缺点是Hibernate Session的Instance可能会大增,使用的JDBC Connection量也会大增,如果Connection Pool的maxPoolSize设得太小,很容易就出问题。此办法放弃。
2.我的resovedQuesWithNoOkAnswer方法没有写权限,这时方法内有update操作,则需要手动设置flush model为Flush.AUTO,如
session.setFlushMode(FlushMode.AUTO); 
session.save(user);
session.flush();

设置后还是出现同样的问题,不知道是不是Session有没有正确的获取到的原因.
3.Extend OpenSessionInViewFilter,Override protected Session getSession(SessionFactory sessionFactory),将FlushMode直接改为Auto。这种方法很麻烦,并且重写后要重新全部编译。
4.检查事务管理配置(applicationContext.xml)
		<property name="transactionAttributes">
<props>
<prop key="save*">PROPAGATION_REQUIRED</prop>
<prop key="update*">PROPAGATION_REQUIRED</prop>
<prop key="delete*">PROPAGATION_REQUIRED</prop>
<prop key="add*">PROPAGATION_REQUIRED</prop>
<prop key="insert*">PROPAGATION_REQUIRED</prop>
<prop key="remove*">PROPAGATION_REQUIRED</prop>
<prop key="set*">PROPAGATION_REQUIRED</prop>
<prop key="login*">PROPAGATION_REQUIRED</prop>
<prop key="*">PROPAGATION_REQUIRED, readOnly</prop>
</props>
</property>

我的方法以resoved开头的不可操作数据库(更新数据库),
它默认进行了PROPAGATION_REQUIRED, readOnly方式处理,我手动再添加一个
			<prop key="resoved*">PROPAGATION_REQUIRED</prop>

重启服务,问题就解决了。
### Java SQL连接只读问题的解决方案 在Java中,当使用`java.sql.Connection`对象执行SQL查询时,如果遇到“connection read-only data modification not allowed”的错误,通常是因为数据库连接被设置为只读模式。以下是解决此问题的详细分析和方法。 #### 1. 检查连接是否为只读 在建立数据库连接时,可以通过`Connection`接口的`isReadOnly()`方法检查当前连接是否处于只读模式[^3]。如果返回值为`true`,则需要调整连接配置以允许写操作。 ```java Connection connection = DbToolkit.getCurrent().getConnection(); if (connection.isReadOnly()) { System.out.println("Connection is in read-only mode."); } ``` #### 2. 设置连接为可写模式 如果确认连接处于只读模式,可以调用`Connection`接口的`setReadOnly(false)`方法将其设置为可写模式[^4]。需要注意的是,某些数据库驱动程序可能不允许更改连接的只读状态,这取决于具体的数据库实现。 ```java try { connection.setReadOnly(false); } catch (SQLException e) { e.printStackTrace(); } ``` #### 3. 检查数据库权限 即使将连接设置为可写模式,如果数据库用户没有足够的权限执行数据修改操作(如`INSERT`、`UPDATE`或`DELETE`),仍然会抛出`SQLException`。因此,需要确保数据库用户具有相应的权限[^5]。 #### 4. 使用事务管理 在执行数据修改操作时,建议显式地管理事务。通过调用`Connection`的`setAutoCommit(false)`方法禁用自动提交,并在操作完成后手动提交或回滚事务[^6]。 ```java try { connection.setAutoCommit(false); // 执行数据修改操作 Statement statement = connection.createStatement(); statement.executeUpdate("UPDATE table_name SET column_name = value WHERE condition"); connection.commit(); } catch (SQLException e) { connection.rollback(); e.printStackTrace(); } finally { connection.setAutoCommit(true); } ``` #### 5. 捕获异常并提供上下文信息 根据引用内容[^2],在捕获`SQLException`时,应尽可能提供详细的上下文信息,以便于定位问题。例如,记录导致异常的SQL语句、参数值以及堆栈跟踪信息。 ```java catch (SQLException e) { System.err.println("Error executing SQL: " + sqlQuery); System.err.println("Parameters: " + Arrays.toString(parameters)); e.printStackTrace(); } ``` ### 示例代码 以下是一个完整的示例,展示了如何解决Java SQL连接只读问题: ```java public void executeUpdate(String sqlQuery, Object[] parameters) throws SQLException { Connection connection = DbToolkit.getCurrent().getConnection(); if (connection.isReadOnly()) { connection.setReadOnly(false); } try { connection.setAutoCommit(false); PreparedStatement statement = connection.prepareStatement(sqlQuery); for (int i = 0; i < parameters.length; i++) { statement.setObject(i + 1, parameters[i]); } statement.executeUpdate(); connection.commit(); } catch (SQLException e) { connection.rollback(); System.err.println("Error executing SQL: " + sqlQuery); System.err.println("Parameters: " + Arrays.toString(parameters)); e.printStackTrace(); throw e; } finally { connection.setAutoCommit(true); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值