无论如何都不是事务性的Spring方法编写

时间:2018-07-11 15:14:39

标签: spring transactions

我在一个项目中有一个主要人员,他实例化了一个名为AnagrafePfDaoImpl(意大利语/英语名称)的DAO。 该DAO具有以下声明:

<bean id="AnagrafePfDaoImpl"
    class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
    <property name="transactionManager">
        <ref bean="transactionManager" />
    </property>
    <property name="target">
        <ref bean="anagraficaDAOTarget" />
    </property>
    <property name="proxyTargetClass" value="true" />
    <property name="transactionAttributes">
        <props>
            <prop key="insert*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="update*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="merge*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="salva*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="delete*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="search*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED</prop>
            <prop key="find*">PROPAGATION_REQUIRED, ISOLATION_READ_COMMITTED,readOnly</prop>
            <prop key="get*">PROPAGATION_REQUIRED,ISOLATION_READ_COMMITTED,readOnly</prop>
        </props>
    </property>
</bean>

我假设,如果我在AnagrafePfDaoImpl中调用名为doS​​tuff()的方法(例如在其中进行更新),则除非从具有自己事务的方法中进行调用,否则这不会产生任何效果。但是我错了,因为如果我从主类(没有声明)中调用doStuff(),则该方法实际上将写入我的数据库中。你能告诉我为什么吗?对不起我的英语不好。 谢谢

1 个答案:

答案 0 :(得分:0)

假定您的组件的定义类似于以下内容:

@Component
public class AnagraficaDAOTarget {

  @Autowired
  DataSource dataSource;

  public long doStuff(String message) throws Exception {
    long id;
    try(Connection connection = dataSource.getConnection()) {
      id = getId(connection);
      try (PreparedStatement statement = connection.prepareStatement("insert into comment (id, message) values (?, ?)")) {
        statement.setLong(1, id);
        statement.setString(2, message);
        int count = statement.executeUpdate();
        if (count != 1) {
          throw new Exception("expected one row insert, actual: " + count);
        }
      }
    }
    return id;
  }
} 

然后,连接处于自动提交模式,因此即使doStuff不是事务性的,更改仍将应用于数据库。