hibernate关于事务和sqlsession的几个坑

我们知道hibernate是ORM关系型数据库。和数据库交互的时候需要sqlsession,如果是保存、更新、删除操作的时候,还需要有事务。

在spring和hibernate整合的时候,事务都是有spring来处理。有的时候会遇到坑。

异常一:

Could not obtain transaction-synchronized Session for current thread

hibernate关于事务和sqlsession的几个坑_第1张图片

之前没有问题的。因为修改了部分代码后出现了这个问题。

经过对比之后,发现正当时情况下:

在service层使用了类级别的事务。使用的是spring的@Transactional注解。

出现异常的时候,是事务注解被注释掉了。

错误代码:

hibernate关于事务和sqlsession的几个坑_第2张图片

使用的是:Session session = getCurrentSession();这个方法。

经过查询得到:

hibernate的sessionFactory中getCurrentSession()方法和OpenSession方法的区别:

hibernate关于事务和sqlsession的几个坑_第3张图片

所以,Could not obtain transaction-synchronized Session for current thread这个异常的原因是因为使用了 getCurrentSession();获取session 而没有使用@Transactional导致的。

扩展:

使用@Transactional默认只有当方法中抛出unchecked的runtimeException时,才会进行回滚,抛出需捕获的Exception异常是不能进行回滚的

hibernate关于事务和sqlsession的几个坑_第4张图片
hibernate关于事务和sqlsession的几个坑_第5张图片
hibernate关于事务和sqlsession的几个坑_第6张图片

异常二:

A different object with the same identifier value was already associated with the session

错误截图:

错误再现:

hibernate关于事务和sqlsession的几个坑_第7张图片

先查询。如果存在,重新set指定字段值之后,调用saveOrUpdateEntity的方法。就出现这个异常。

根据错误提示,可以分析得到:

在同一个session中,存在两个相同的标识(如主键id),但是这两个实体有不是相同的。

虽然使用了:

BaseNewinfoViewTotal entity =model.map(bean,BaseNewinfoViewTotal.class);

其实质也相当于是new了一个BaseNewinfoViewTotal 对象。然后重新赋值而已。

所以,两个对象都有相同的主键ID,但是在内存中是两个不同的对象。

这个时候解决方案:

采用session.merge (object c)代替session.save(object c),即可解决

hibernate关于事务和sqlsession的几个坑_第8张图片

你可能感兴趣的:(hibernate关于事务和sqlsession的几个坑)