4

我有两种方法可以执行以下操作:

void withdraw(int amount) {
    transactionTemplate.execute(new TransactionCallback() {
        @Override
        public Object doInTransaction() { ... }
    }
}

void deposit(int amount) {
    transactionTemplate.execute(new TransactionCallback() {
        @Override
        public Object doInTransaction() { ... }
    }
}

现在我想定义在同一个事务中同时执行的第三种方法:

void transferTo(Account other, int amount) {
    transactionTemplate.execute(new TransactionCallback() {
        @Override
        public Object doInTransaction() {
            withdraw(amount);
            other.deposit(amount);
            return null;
        }
    }
}

事务传播设置为 REQUIRED(默认)。

用例很明显,如果独立调用,我希望withdraw() 和 deposit() 在它们自己的事务中运行。但是,如果我调用 transfer(),我希望所有语句都在同一个事务中运行。

但是,当我运行 transfer() 时,出现以下异常:

Caused by: java.lang.IllegalStateException: Already value [org.springframework.jdbc.datasource.ConnectionHolder@e4b2ed] for key [org.apache.tomcat.dbcp.dbcp.BasicDataSource@26cc37d2] bound to thread [http-bio-8084-exec-13]
    at org.springframework.transaction.support.TransactionSynchronizationManager.bindResource(TransactionSynchronizationManager.java:189)
    at org.springframework.jdbc.datasource.DataSourceTransactionManager.doBegin(DataSourceTransactionManager.java:234)
4

1 回答 1

3

好吧,我发现了问题...并不是真正解决我的问题,而是解释了为什么我会收到错误。

我使用的事务管理器是 org.springframework.ldap.transaction.compensating.manager.ContextSourceAndDataSourceTransactionManager因为我还需要执行一些 LDAP 相关的操作。

唯一的问题是ContextSourceAndDataSourceTransactionManager不支持嵌套事务。

因此嵌套的 transactionTemplate 调用将失败......不方便。

编辑:进一步调查表明ContextSourceTransactionManager也不支持嵌套事务。

于 2013-04-15T17:08:58.857 回答