使用谷歌应用引擎正确处理交易回滚

时间:2011-05-30 05:47:52

标签: java google-app-engine

我不确定我是否理解谷歌文档,我想知道是否有其他人可以检查我的理解。

以下代码是否保证只执行以下两项操作:

  • 更新到银行帐户余额
  • 存储交易记录。

以下是我的理解是正确的:

public void add(String account, Double value, String description, Date date) throws EntityNotFoundException {
    DatastoreService datastore = DatastoreServiceFactory.getDatastoreService();
    int retries = 3;
    while (true) {
        Transaction txn = datastore.beginTransaction();
        try {

            // Update the bank balance
            Key key = KeyFactory.createKey("Account", account);
            Entity e = datastore.get(key);
            Double balance = (Double) e.getProperty("balance");
            balance += value;
            e.setProperty("balance", value);
            datastore.put(e);

            // Record transaction details
            Entity d = new Entity("Transaction", key);
            d.setProperty("account_key", key);
            d.setProperty("date", date);
            d.setProperty("value", value);
            d.setProperty("description", description);

            txn.commit();
            break;
        } catch (ConcurrentModificationException e) {
            if (retries == 0) throw e;
            retries--;
        } finally {
            if (txn.isActive()) txn.rollback();
        }
    }
    }
}

2 个答案:

答案 0 :(得分:2)

这是对的。但是,没有必要在实体中包含帐户的密钥 - 您创建的实体是相关帐户的子项。

答案 1 :(得分:0)

不一定是ConcurrentModificationException指示事务失败。文档http://code.google.com/appengine/docs/java/datastore/transactions.html中的绝对歧义是“如果您的应用在提交事务时收到异常,则并不总是意味着事务失败。您可以在以下情况下收到DatastoreTimeoutException,ConcurrentModificationException或DatastoreFailureException异常事务已经提交并最终成功应用。尽可能使数据存储区事务处于幂等状态,这样如果重复事务,最终结果将是相同的。“ 我已经在Transactions and ConcurrentModificationException documentation ambiguity为此问题创建了一个单独的主题,因此如果有人了解它应该如何处理,请同时更新。