我有一个SLSB,它可以递增实体中的数字。如果两个线程同时到达 SLSB,则我在两个请求中得到相同的数字。
SLSB提取物
@Stateless(mappedName = "ejb/CustomerManager")
public class CustomerManagerBean implements CustomerManager {
...
public String recoverName(int id) {
Customer customer = (Customer) em.createQuery("from Customer where id = :id").setParameter("id", id).getSingleResult();
int oldValue = customer.getValue();
int newValue = oldValue + 1;
customer.setValue(newValue);
[BP] return customer.getName() + " value=" + customer.getValue();
}
...
}
实体提取
@Entity
public class Customer implements Serializable {
@Id
private int id;
private int value;
}
为了测试问题,我在SLSB recoverName方法中标有[BP]
的行处有一个断点。然后从两个单独的浏览器页面进行两次调用。在断点处,两个调用的值相同。
当第二个调用尝试使用setter修改值时,它不应该引发某种异常吗?
我使用 JBoss 5 作为 AS,使用 MySql 或 Oracle 作为数据库(两者都尝试过)
感谢您的帮助
如果向实体添加@Version注释字段,则会在刷新时出现异常,JPA 将使用该字段进行乐观锁定。
每次 JPA 更新实体时,它都会将内存中的版本与数据库中的版本进行比较,如果它们不匹配,则会引发异常。如果它们匹配,则版本将递增。
只需将以下内容添加到您的实体:
@Column(name = "version")
@Version
private long version;
(当然,并将相应的列添加到数据库中)