为什么Hibernate会抛出org.hibernate.exception.LockAcquisitionException?

2022-09-04 02:10:57

我有这个方法:

mymethod(long id){  
    Person p = DAO.findPerson(id);

    Car car = new Car();
    car.setPerson(p);
    p.getCars().add(car);

    DAO.saveOrUpdate(car);
    DAO.saveOrUpdate(p);
    DAO.delete(p.getCars().get(0));//A person have many cars
}  

映射:

人.hbm.xml

<!-- one-to-many : [1,1]-> [0,n] -->
<set name="car" table="cars" lazy="true" inverse="true">
    <key column="id_doc" />
    <one-to-many class="Car" />
</set>

<many-to-one name="officialCar"
class="Car" 
column="officialcar_id" lazy="false"/>  

Cars.hbm.xml

<many-to-one name="person" class="Person"
            column="id_person" not-null="true" lazy="false"/>   

此方法适用于单个线程,并且在多个线程上,会给我一个错误:

02/08/2014 - 5:19:11 p.m. - [pool-1-thread-35] - WARN - org.hibernate.util.JDBCExceptionReporter - SQL Error: 60, SQLState: 61000 
02/08/2014 - 5:19:11 p.m. - [pool-1-thread-35] - ERROR - org.hibernate.util.JDBCExceptionReporter - ORA-00060: deadlock detection while waiting for a resource 
 
02/08/2014 - 5:19:11 p.m. - [pool-1-thread-35] - WARN - org.hibernate.util.JDBCExceptionReporter - SQL Error: 60, SQLState: 61000 
02/08/2014 - 5:19:11 p.m. - [pool-1-thread-35] - ERROR - org.hibernate.util.JDBCExceptionReporter - ORA-00060: deadlock detection while waiting for a resource 
 
02/08/2014 - 5:19:11 p.m. - [pool-1-thread-35] - ERROR - org.hibernate.event.def.AbstractFlushingEventListener - Could not synchronize database state with session 
org.hibernate.exception.LockAcquisitionException: Could not execute JDBC batch update

AOP 交易 :

<tx:advice id="txAdviceNomService" transaction-manager="txManager">
    <tx:attributes>
        <tx:method name="*" propagation="REQUIRED" rollback-for="java.lang.Exception" />
        <tx:method name="getAll*" read-only="true" propagation="SUPPORTS" />
        <tx:method name="find*" read-only="true" propagation="SUPPORTS" />
    </tx:attributes>
</tx:advice>

注意:当我在更新后添加Thread.sleep(5000)时,没关系。但是这个解决方案并不干净。


答案 1

根据您的映射,操作序列应如下所示:

Person p = DAO.findPerson(id);

Car car = new Car();
car.setPerson(p);

DAO.saveOrUpdate(car);

p.getCars().add(car);

Car firstCar = p.getCars().get(0);
firstCar.setPerson(null);
p.getCars().remove(firstCar);
if (p.officialCar.equals(firstCar)) {
   p.officialCar = null;
   p.officialCar.person = null;
}

DAO.delete(firstCar);

更新或删除意味着获取独占锁,即使在READ_COMMITTED隔离级别也是如此。

如果另一个事务想要使用当前正在运行的事务(该事务已锁定有问题的此行)更新同一行,则不会出现死锁,而是锁定获取超时异常。

由于您遇到了死锁,这意味着您在多个表上获取了锁,并且未正确排序锁的获取。

因此,请确保服务层方法设置事务边界,而不是 DAO 方法。我看到你声明了 get and find 方法来使用 SUPPORTED,这意味着只有当当前启动一个事务时,它们才会使用事务。我认为您也应该为这些使用必需的,但只需将它们标记为.read-only = true

因此,请确保事务方面将事务边界应用于“mymethod”,而不是DAO。


答案 2

我有汽车 -> (1 -n) 的地方。我在桌子上有一个外键(id_car)。此外键没有索引。当我向此外键添加索引时,我的问题已解决。

请参阅此答案


推荐