浏览 2025 次
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
|
|
---|---|
作者 | 正文 |
发表时间:2011-01-10
最后修改:2011-11-07
有一个持久化类A需要包括n个有序的long型数。类A需要保存到数据库中。 类A如下: public class Foo{ // some properties List longNumbers=new ArrayList(); // some propertes' getter and setter public List getLongNumbers(){ return this.longNumbers; } public void setLongNumber(List longNubmers){ this.longNumbers=longNumbers; } } 这里仅讨论值集合对应的hibernate配置如下: <list name="longNumbers" table="itil_foolongnumbers" lazy="false" inverse="false" cascade="all-delete-orphan" > <key column="ifooid" > </key> <index column="iindex" /> <element column="ilongnumber" type="java.lang.Long" not-null="false" unique="false" /> </list> 配置好后,就可以直接向Foo对象里添加long型的值了,并通过hibernate的Session进行增删改查了。 但是,我遇到了一个问题。当同时有两个或两个以上线程执行如下代码时出现错误。(代码只表语意并不严格) public void appdendLongNumber(Long fooId,Long num){ Transaction tr=session.beginTransaction(); Foo f=session.get(fooId,Foo.class); f.getLongNumbers().add(num); tr.commit(); ... } 出现的错误并不是hibernate报错,而是数据库报主键冲突异常。 经过分析发现:原来itil_foolongnumbers表是用ifooid和iindex做为复合主键的。当有两个线程同时向OID相同的Foo对象增加新的Long Number时,就使得在两个线程中的Foo对象分别拥有了具有相同下标的Long Number元素。然后,先被保存的Foo对象成功,第二个保存到数据库的,就引发主键冲突了。 这个问题在开发时是很难想到的。 这个问题,可以按如下方式解决。 两个方法: 1.代码级互斥。 2.重试。 声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2011-01-10
主键用uuid
|
|
返回顶楼 | |
发表时间:2011-01-10
tanqimin 写道 主键用uuid
hiberante的List值类型集合,能用uuid做主键吗? |
|
返回顶楼 | |