记得之前被 session的问题困扰了很久,也不断遇到很多问题,自己也提出许多解决方案,认识在不断加深中....
我把笔记发上来
执行更新update操作时候出现的问题,后台采用hibernateTemplate操作
Write operations are not allowed in read-only mode (FlushMode.NEVER) - turn your Session into FlushMode.AUTO or remove 'readOnly' marker from transaction definition
OpenSessionInViewFilter在getSession的时候,会把获取回来的session的flush mode 设为FlushMode.NEVER。然后把该sessionFactory绑定到TransactionSynchronizationManager,使request的整个过程都使用同一个session,在请求过后再接除该sessionFactory的绑定,最后closeSessionIfNecessary根据该session是否已和transaction绑定来决定是否关闭session。在这个过程中,若HibernateTemplate 发现自当前session有不是readOnly的transaction,就会获取到FlushMode.AUTO Session,使方法拥有写权限。
也即是,如果有不是readOnly的transaction就可以由Flush.NEVER转为Flush.AUTO,拥有insert,update,delete操作权限,如果没有transaction,并且没有另外人为地设flush model的话,则doFilter的整个过程都是Flush.NEVER。所以受transaction保护的方法有写权限,没受保护的则没有。
所以说当配制有OpenSessionInViewFilter就会出现readOnly问题,而不配又出现session关闭的问题,要用hibernateTemplate的话就要考虑配置spring事务
如果用hibernate (uuid.hex)主键生成方式,用spring getSession.save()方法 会插不了数据
用hibernateTemplate就可以
这样的话配置了OpenSessionInViewFilter就出现问题了
想了很久都不知道怎么做,网上总说用事务解决,但我总没成功,
现在其他方法都没问题,就单一个插入操作 我用this.getHibernateTemplate().save();就出现
Write operations are not allowed in read-only mode (FlushMode.NEVER) - turn your Session into FlushMode.AUTO or remove 'readOnly' marker from transaction definition
最后没办法,就单单2个插入操作不可能去改变表结构或者动用其他东西
OpenSessionInViewFilter这家伙不配可不行,我自己用了一个解决方案
public void insert(Consumer c) throws Exception {
final Consumer cc=c;
// this.getHibernateTemplate().save(c);有事务问题
// this.getSession().save(c);
// super.getSession(true).save(c);
// this.getSession().saveOrUpdate(c);这些都插不了数据,也没报错
this.getHibernateTemplate().execute(
new HibernateCallback() {
public Object doInHibernate(Session session) throws HibernateException {
session.setFlushMode(FlushMode.AUTO);
session.save(cc);
session.flush();
return null;
}
});
}