0

我创建了一个实体用户,它具有整数属性交易配额。我需要根据其价值更新交易配额。

如果大于 0,则需要递减,否则需要保留其值。

所以这是我的算法。

  1. 从数据库中检索用户实体。
  2. 检查事务配额,如果事务配额大于0,则减一。
  3. 保留更改的用户实体。

在上述情况下,当并发请求出现时,如果两个线程检索相同的用户实体,然后两个线程都读取相同的事务配额值,如果它大于 0,则都减 1 并更新用户实体。

前任。

ThreadA: val = e.getTxnQuota(); val = 5
ThreadB: val = e.getTxnQuota(); val = 5
ThreadA: e.setTxnQuota(val- 1); val = 4 
ThreadB: e.setTxnQuota(val- 1); val =4
ThreadA: eDao.save(e);
ThreadB: eDao.save(e);

在上述情况下,保存的值是 4 而不是 3。

那么有什么方法可以创建一个原子事务,我可以在其中检查事务配额并更新用户实体?

4

1 回答 1

1

这称为锁定,通常最好使用乐观锁定。JPA 附带了一个标准机制来执行此操作。只需向您的实体添加一个版本字段,并使用@Version 对其进行注释:

@Version
private long version;

瞧!

每次保存用户时,其当前版本将与数据库中的版本进行比较,并递增:

update user set ..., version = version + 1 where id = ... and version = theVersionOfTheUserWhenItWasLoaded

(这一切都是透明地发生的)。

如果版本不匹配,则不会更新任何内容,JPA 引擎会检测到它,抛出 OptimisticLockException,并将事务标记为回滚。

旁注:在上面的场景中,调用save()是不必要的:对附加实体所做的所有更改都会自动保存在数据库中。无需保存实体。

于 2012-05-25T17:17:01.080 回答