Java Hibernate get方法返回线程中记录的旧版本
我们有java web应用程序,我们使用的是spring、hibernate3,我在从DB获取记录的实际版本时遇到了问题 以下是代码片段:Java Hibernate get方法返回线程中记录的旧版本,java,spring,hibernate,web-applications,Java,Spring,Hibernate,Web Applications,我们有java web应用程序,我们使用的是spring、hibernate3,我在从DB获取记录的实际版本时遇到了问题 以下是代码片段: Class Scheduler { public void scheduleTests(………….) { . . . TestCase testCase; . //Updating entity getHibernateTemplate ().saveOrUpdate(testCase)
Class Scheduler
{
public void scheduleTests(………….)
{
.
.
.
TestCase testCase;
.
//Updating entity
getHibernateTemplate ().saveOrUpdate(testCase);
getHibernateTemplate ().flush();
.
.
.
Starts new thread
.
}
Thread run{
.
.
TestCase testCase = getTestCase(testCaseId);
.
.
.
.
updateTestCase(testCase);
}
getTestCase(testCaseId)
{
TestCase testCase = null;
try
{
testCase=(TestCase)getHibernateTemplate().get(TestCase.class, testCaseId);
}
catch (Exception ex)
{
ex.printStackTrace();
}
return testCase;
}
updateTestCase(TestCase testCase)
{
try
{
getHibernateTemplate ().saveOrUpdate(testCase);
getHibernateTemplate ().flush();
}
catch (Exception ex)
{
ex.printStackTrace();
}
}
在MethodScheduleTest中,我更改了对象TestCase,并将其保存到DB中。然后新线程开始,在这个线程的开头,我向DB询问来自DB的TestCase的实际版本。然后执行一些功能,最后将TestCase的状态更新为DB
问题是当我运行这个场景超过1次时。然后奇怪的事情发生了。在线程方法运行中,我从方法getTestCase获得旧版本的TestCase,然后更新崩溃,出现以下异常:
错误
org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:300) - Could not synchronize database state with session
"org.hibernate.StaleObjectStateException: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect): [at.h3g.portal.db.webshop.WsRcCheck#405]
at org.hibernate.persister.entity.AbstractEntityPersister.check(AbstractEntityPersister.java:1680)
at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:2271)
at org.hibernate.persister.entity.AbstractEntityPersister.updateOrInsert(AbstractEntityPersister.java:2180)
at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:2456)
at org.hibernate.action.EntityUpdateAction.execute(EntityUpdateAction.java:91)
at org.hibernate.engine.ActionQueue.execute(ActionQueue.java:248)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:232)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:140)
at org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:297)
at org.hibernate.event.def.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:27)
at org.hibernate.impl.SessionImpl.flush(SessionImpl.java:988)
at org.springframework.orm.hibernate3.HibernateTemplate$29.doInHibernate(HibernateTemplate.java:815)
at org.springframework.orm.hibernate3.HibernateTemplate.execute(HibernateTemplate.java:366)
at org.springframework.
ERROR org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:300) - Could not synchronize database state with session
"org.hibernate.StaleObjectStateException: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect): [at.h3g.portal.db.webshop.WsRcCheck#405]
at org.hibernate.persister.entity.AbstractEntityPersister.check(AbstractEntityPersister.java:1680)
at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:2271)
at org.hibernate.persister.entity.AbstractEntityPersister.updateOrInsert(AbstractEntityPersister.java:2180)
at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:2456)
at org.hibernate.action.EntityUpdateAction.execute(EntityUpdateAction.java:91)
at org.hibernate.engine.ActionQueue.execute(ActionQueue.java:248)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:232)
at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:140)
at org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:297)
at org.hibernate.event.def.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:27)
at org.hibernate.impl.SessionImpl.flush(SessionImpl.java:988)
at org.hibernate.impl.SessionImpl.managedFlush(SessionImpl.java:337)
at org.hibernate.transaction.JDBCTransaction.commit(JDBCTransaction.java:106)
at org.springframework.orm.hibernate3.HibernateTransactionManager.doCommit(HibernateTransactionManager.java:584)
at org.springframework.transaction.support.AbstractPlatformTransactionManager.processCommit(AbstractPlatformTransactionManager.java:500)
at org.springframework.transaction.support.AbstractPlatformTransactionManager.commit(AbstractPlatformTransactionManager.java:473)
at org.springframework.transaction.interceptor.TransactionAspectSupport.doCommitTransactionAfterReturning(TransactionAspectSupport.java:267)
at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:106)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:170)
为什么我得到了旧版本的TestCase?我能做什么?
下面是Hibernate和Spring的配置
在Web.Xml中:
<filter>
<filter-name>hibernateFilter</filter-name>
<filter-class>org.springframework.orm.hibernate3.support.OpenSessionInViewFilter</filter-class>
<init-param>
<param-name>sessionFactoryBeanName</param-name>
<param-value>sessionFactory</param-value>
</init-param>
</filter>
冬眠过滤器
org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
sessionFactoryBeanName
会话工厂
和上下文配置:
<bean id="sessionFactory"
class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource">
<ref bean="dataSource" />
</property>
<property name="mappingResources">
<list>
.
.
.
</list>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">
org.hibernate.dialect.Oracle9Dialect
</prop>
<prop key="hibernate.connection.release_mode">
auto
</prop>
</props>
</property>
</bean>
<bean id="dataSource"
class="org.springframework.jndi.JndiObjectFactoryBean">
<property name="jndiName">
<value>java:comp/env/jdbc/SOMEDATASOURCE</value>
</property>
</bean>
<bean id="hibernateTemplate"
class="org.springframework.orm.hibernate3.HibernateTemplate">
<property name="sessionFactory">
<ref bean="sessionFactory" />
</property>
</bean>
<bean id="transactionManager"
class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory">
<ref local="sessionFactory" />
</property>
</bean>
<bean id="transactionInterceptorTestScheduling"
class="org.springframework.transaction.interceptor.TransactionInterceptor">
<property name="transactionManager">
<ref bean="transactionManager" />
</property>
<property name="transactionAttributeSource">
<value> org.company.db.Scheduler.updateTestCase=PROPAGATION_REQUIRES_NEW
</value>
</property>
</bean>
.
.
.
org.hibernate.dialogue.oracle9dialogue
汽车
java:comp/env/jdbc/SOMEDATASOURCE
org.company.db.Scheduler.updateTestCase=PROPAGATION\u REQUIRES\u NEW
我想补充一点,那个时只创建了一个线程,并没有其他线程访问数据库中的测试用例数据。唯一的问题是getTestCase返回旧版本的TestCase记录,我现在不知道为什么
谢谢您的建议。问题在于hibernate会话已绑定到当前线程。所以一个解决方案是不要使用线程,或者应该创建事务拦截器,它将负责线程的会话。更多信息请参见仅猜测。尝试通过单独的事务包装保存和获取。。。