我创建了一个实体用户,它具有整数属性事务配额。我需要根据其值更新交易配额。
如果它大于 0,则需要递减,否则需要保留其值。
这是我的算法。
- 从数据库中检索用户实体。 检查事务配额,
- 如果事务配额大于 0,则将其减少 1。
- 保留更改的用户实体。
在上述情况下,当并发请求出现时,如果两个线程检索相同的用户实体,并且然后,两个线程读取相同的事务配额值,如果它大于 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。
那么,有什么方法可以创建一个原子事务,在其中我可以检查事务配额并更新用户实体?
这称为锁定,通常建议使用乐观锁定。JPA 带有一个标准机制来做到这一点。只需向实体添加一个版本字段,并使用@Version对其进行批注:
@Version
private long version;
瞧!
每次保存用户时,其当前版本将与数据库中的版本进行比较,并递增:
update user set ..., version = version + 1 where id = ... and version = theVersionOfTheUserWhenItWasLoaded
(这一切都是透明地发生的)。
如果版本不匹配,则不会更新任何内容,JPA 引擎会检测到它,抛出 OptimisticLockException,并将事务标记为回滚。
旁注:在上面的方案中,不需要调用save()
:对附加实体所做的所有更改都会自动保存在数据库中。无需保存实体。