问题现象:
第一个用户投票不成功后,换第二个用户登录,然后投票,投票成功,投票数加1(由14票,变更为15票)。这时刷新页面发现投票数为未投票前的数(14票),再刷新又为更新后的值(15票),不停的刷新,投票数会在新旧值(14与15)之间交替出现。
查看数据库中的表,值是最新的15票。
通过跟踪程序发现,当投票不成功时事务没有正常关闭。代码如下:
session由 spring的工厂BEAN 址 org.springframework.orm.hibernate3.LocalSessionFactoryBean生成。
当事务打开的时候,会去线程的上下文LocalThread中找SessionHolder,如果没有找到就打开一个新的Session。包装好,放到LocalThread中去。
当事务关闭的时候,会把LocalThread中的对象清空,当前线程持有的Session关闭。
(spring中由TransactionSynchronizationManager维护当前线程的session 及事务信息)
如果事务没有关闭,会导致session一直在当前线程的LocalThread中。当下次请求到过中间件的时候,由于中间件有线程池,可能会从池中调用上次持有Session的线程
来处理新的请求,当Session要加载对象时,会从session级的缓存中查找,找到后就直接返回了,这时就看到原来的旧值。
使用spring的HibernateTemplate时,由于当前SESSION中存在事务,在执行完doInHibernate后并不会关闭session,由此会引起一系统的问题,比如session中缓存的对象会越来越多。
第一个用户投票不成功后,换第二个用户登录,然后投票,投票成功,投票数加1(由14票,变更为15票)。这时刷新页面发现投票数为未投票前的数(14票),再刷新又为更新后的值(15票),不停的刷新,投票数会在新旧值(14与15)之间交替出现。
查看数据库中的表,值是最新的15票。
通过跟踪程序发现,当投票不成功时事务没有正常关闭。代码如下:
TransactionStatus status = [color=Red]beginTransaction();[/color]
try {
if (list != null) {
for (int i = 0; i < list.size(); i++) {
if (((JczdTpWt) list.get(i)).getNRybh() == userid) {
return "fail";<!-- [color=Red]问题就在这 事务没有正常关闭就退出了[/color]。-->
}
}
}
JczdDa jczdDa = dao.getDaById(daid);
jczdDa.setNTpsl(jczdDa.getNTpsl() + 1);
dao.updateDa(jczdDa);
。。。省略
[color=Red]getTransactionManager().commit(status);[/color]
session由 spring的工厂BEAN 址 org.springframework.orm.hibernate3.LocalSessionFactoryBean生成。
当事务打开的时候,会去线程的上下文LocalThread中找SessionHolder,如果没有找到就打开一个新的Session。包装好,放到LocalThread中去。
当事务关闭的时候,会把LocalThread中的对象清空,当前线程持有的Session关闭。
(spring中由TransactionSynchronizationManager维护当前线程的session 及事务信息)
如果事务没有关闭,会导致session一直在当前线程的LocalThread中。当下次请求到过中间件的时候,由于中间件有线程池,可能会从池中调用上次持有Session的线程
来处理新的请求,当Session要加载对象时,会从session级的缓存中查找,找到后就直接返回了,这时就看到原来的旧值。
使用spring的HibernateTemplate时,由于当前SESSION中存在事务,在执行完doInHibernate后并不会关闭session,由此会引起一系统的问题,比如session中缓存的对象会越来越多。