Write operations are not allowed in read-only mode (FlushMode.MANUAL): Turn your Session into FlushM

本文介绍了在使用Spring和Hibernate框架时遇到的数据保存异常问题,该异常源于默认的只读模式设置。文中提供了详细的配置调整步骤,包括事务管理和OpenSessionInViewFilter的配置,最终通过修改FlushMode解决了问题。

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

当我写数据的时候出现上面异常,查啊查,应该是保存数据时采用了默认的只读模式。

网上的方法配置事务:

1:

<!-- 定义事务管理 -->
    <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory" />
    </bean>
    <!-- 配置事务异常封装 -->
   <bean id="persistenceExceptionTranslationPostProcessor" 
       class="org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor" />

   <!--  声明式容器事务管理 ,transaction-manager指定事务管理器为transactionManager -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="add*" propagation="REQUIRED" />
            <tx:method name="get*" propagation="REQUIRED" />
            <tx:method name="*" read-only="true" />
        </tx:attributes>
    </tx:advice>

    <aop:config expose-proxy="true">
        <!-- 只对业务逻辑层实施事务 -->
        <aop:pointcut id="txPointcut" expression="execution(* com.zq.app.cms.service..*.*(..))" />
        <!-- Advisor定义,切入点和通知分别为txPointcut、txAdvice -->
        <aop:advisor pointcut-ref="txPointcut" advice-ref="txAdvice"/>
    </aop:config>

2.修改默认只读模式:

<filter>
      <filter-name>OpenSessionInViewFilter</filter-name>
      <filter-class>org.springframework.orm.hibernate4.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>
    <filter-mapping>
        <filter-name>OpenSessionInViewFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

都没用。费解。。。

我采用的是spring 4.20 hibernate 4.3.10.Final

把 penSession()方法中 session.setFlushMode(FlushMode.MANUAL); 改为session.setFlushMode(FlushMode.AUTO);
结果是可以的。

### 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、付费专栏及课程。

余额充值