假设我有以下签名的方法
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
}
答案 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
中另一个事务完成的提取和锁定数据之间可能会发生。
请注意,getData
和updateData
中使用的实体经理必须相同。