JPA中的并发(陈旧数据)问题

时间:2010-10-20 08:02:56

标签: java orm jpa

假设我有以下签名的方法

Object getData(int id) {  
  //create a entity manager
  //get data frm db
  //return data
}

updateData() {
  Object obj = getData(id) 
  //get entity manager
  //start transcation tx
  //update
  //commit tx
}

现在会引起并发问题吗?在最坏的情况下,数据是否会过时?例如。     如果我getData,并且在我更新时,如果有人更新数据,我的updateData会有陈旧数据吗? 现在我可以使用以下内容:我能解决问题吗?

Object getData(int id,Entitymanager em) {  

      //get data frm db using em
      //return data
    }

 updateData() {
      Object obj = getData(id) 
      //get entity manager em
      //start transcation tx
      //getdata using getData(id,em)
      //commit tx
    }

3 个答案:

答案 0 :(得分:6)

是的,这可能发生。

如果您获得实体(版本1),其他人修改它(创建版本2),然后您修改版本1并保存它,版本2中的任何更改都将丢失。

要阻止这种情况发生,请通过向实体添加@Version属性来使用乐观并发。如果在get和update之间发生了提交,则会抛出异常。在这一点上,您可以选择最佳选择来处理它。

答案 1 :(得分:1)

在多线程环境中也可能存在问题,如果同时多次访问同一段代码,它可以创建锁,为此你可以使用行级锁定技术,这将有助于你。检查出来以下链接。

http://download.oracle.com/docs/cd/B19306_01/server.102/b14220/consist.htm

答案 2 :(得分:1)

如果两个单独的请求同时访问updateData(),您可能会收到陈旧数据。您可以通过锁定updateData()中提取的数据来处理陈旧性。如果您使用Hibernate作为JPA提供程序,则可以按如下方式锁定数据:

updateData() {
    Object obj = getData(id);
    Session session = (Session) em.getDelegate();
    session.refresh(obj, LockMode.UPGRADE);
}

刷新是必要的,因为在updateData中另一个事务完成的提取和锁定数据之间可能会发生。

请注意,getDataupdateData中使用的实体经理必须相同。