Spring Hibernate-无法获得当前线程的事务同步会话


106

我使用spring + hibernate创建了一个应用程序,但始终会收到此错误。这是我第一个使用hibernate的应用程序,我阅读了一些指南,但无法解决此问题。我在哪里做错了?

这是我的应用程序的代码

ott 05, 2014 4:03:06 PM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
Informazioni: Refreshing   org.springframework.context.support.ClassPathXmlApplicationContext@1eab16b: startup date  [Sun Oct 05 16:03:06 CEST 2014]; root of context hierarchy
ott 05, 2014 4:03:06 PM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
Informazioni: Loading XML bean definitions from class path resource [springConfig.xml]
ott 05, 2014 4:03:08 PM org.hibernate.annotations.common.reflection.java.JavaReflectionManager <clinit>
INFO: HCANN000001: Hibernate Commons Annotations {4.0.5.Final}
ott 05, 2014 4:03:08 PM org.hibernate.Version logVersion
INFO: HHH000412: Hibernate Core {4.3.6.Final}
ott 05, 2014 4:03:08 PM org.hibernate.cfg.Environment <clinit>
INFO: HHH000206: hibernate.properties not found
ott 05, 2014 4:03:08 PM org.hibernate.cfg.Environment buildBytecodeProvider
INFO: HHH000021: Bytecode provider name : javassist
ott 05, 2014 4:03:09 PM org.hibernate.dialect.Dialect <init>
INFO: HHH000400: Using dialect: org.hibernate.dialect.MySQL5Dialect
ott 05, 2014 4:03:09 PM org.hibernate.engine.transaction.internal.TransactionFactoryInitiator initiateService
INFO: HHH000399: Using default transaction strategy (direct JDBC transactions)
ott 05, 2014 4:03:09 PM org.hibernate.hql.internal.ast.ASTQueryTranslatorFactory <init>
INFO: HHH000397: Using ASTQueryTranslatorFactory
Exception in thread "main" org.hibernate.HibernateException: Could not obtain transaction-synchronized Session for current thread
at org.springframework.orm.hibernate4.SpringSessionContext.currentSession(SpringSessionContext.java:134)
at org.hibernate.internal.SessionFactoryImpl.getCurrentSession(SessionFactoryImpl.java:1014)
at coreservlets.StudentDAOImpl.create(StudentDAOImpl.java:19)
at coreservlets.MainApp.main(MainApp.java:14)

学生.java

package coreservlets;

public class Student {

    private Integer id;
    private String name;
    private Integer age;

    public Integer getId(){return id;}//getId

    public void setId(Integer id){this.id=id;}//setId

    public String getName(){return name;}//getName

    public void setName(String name){this.name=name;}//setName

    public Integer getAge(){return age;}//getAge

    public void setAge(Integer age){this.age=age;}//setAge

}//Student

学生DAO.java

package coreservlets;

import org.hibernate.SessionFactory;

public interface StudentDAO {

    public void setSessionFactory(SessionFactory sessionFactory);

    public void create(String name,Integer age);

}//StudentDAO

StudentDAOImpl.java

package coreservlets;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Repository;

@Repository
public class StudentDAOImpl implements StudentDAO {

    private SessionFactory sessionFactory;

    @Autowired
    public void setSessionFactory(SessionFactory sessionFactory){
        this.sessionFactory=sessionFactory;
    }//setSessionFactory

    public void create(String name,Integer age){
        Session session=sessionFactory.getCurrentSession();
        Student student=new Student();
        student.setName(name);
        student.setAge(age);
        session.save(student);
    }//create

}//StudentDAOImpl

MainApp.java

package coreservlets;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MainApp {

    public static void main(String[] args) {

        ApplicationContext context=new ClassPathXmlApplicationContext("springConfig.xml");

        StudentDAOImpl student=(StudentDAOImpl) context.getBean("studentDAOImpl");

        student.create("Alessandro", new Integer(33));


    }//main

}//MainApp

springConfig.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
    http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.1.xsd">

<context:annotation-config/>

<context:component-scan base-package="coreservlets"/>

<bean id="dataSource" class="org.apache.commons.dbcp2.BasicDataSource">
  <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
  <property name="url" value="jdbc:mysql://localhost:3306/spring_hibernate"/>
  <property name="username" value="root"/>
  <property name="password" value="password"/>
  <property name="initialSize" value="5"/>
  <property name="maxTotal" value="10"/>
</bean>

<bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="hibernateProperties">
    <value>
            hibernate.dialect=org.hibernate.dialect.MySQLDialect
    </value>
</property>

</bean>

</beans>

sql

create table student
(
id integer not null auto_increment,
name varchar(20) not null,
age integer not null,
primary key(id)
);

3
您是否尝试过将@Transactional添加到您的DAO创建方法中?
约翰·

1
您忘记了声明一个HibernateTransactionManager,并使该方法使用Hibernate事务性。
JB Nizet 2014年

@itachi不正确,sessionFactory.openSession()交易将被禁用。因为它们不是同一会话。>在类服务@Patrikoko中添加spring的注释@Transactional是正确的!看到这个问题:stackoverflow.com/questions/15620355/…示例:@Transactional(readOnly = true, propagation = Propagation.REQUIRED, rollbackFor = {java.lang.Exception.class})
nvnpnco

Answers:


200

您必须启用交易支持(<tx:annotation-driven>@EnableTransactionManagement)并声明transactionManager并且它应通过进行工作SessionFactory

您必须添加@Transactional到您的@Repository

随着@Transactional在你的@Repository春天是能够应用的事务处理支持到你的资料库。

您的Student类没有@ javax.persistence。*注释how @Entity,我假设该类的映射配置已通过XML定义。


1
拜托,您能写应用程序的代码,因为它不起作用吗。。这是我的第一个Hibernate应用程序
Alex

3
<tx:annotation-driven>的等效注释为@EnableTransactionManagement
Anand Rockzz

5
另外,请确保您使用的是org.springframework.transaction.annotation.Transactional,而不是javax.persistance.Transactional
imnd_neel

干杯,我不敢相信我错过了这个注释:)。
Boldbayar '16

1
我已经尝试了几个小时才能使事务正常运行,最后我使用@EnableTransactionManagement代替了<tx:annotation-driven>,并且一切正常。我不能感谢你,曼努埃尔
阿布·苏莱曼,

38

我遇到过同样的问题,但是在一个不属于服务层的类中。在我的情况下,通过该getBean()方法可以从上下文简单地获得事务管理器,并且该类属于视图层-我的项目利用了OpenSessionInView这种技术。

sessionFactory.getCurrentSession()方法已引起与作者相同的异常。对我来说解决方案很简单。

Session session;

try {
    session = sessionFactory.getCurrentSession();
} catch (HibernateException e) {
    session = sessionFactory.openSession();
}

如果该getCurrentSession()方法失败,则openSession()应该解决问题。


从Hibernate3升级到Hibernate5时,我不得不将代码从SessionFactoryUtils.getSession()更改为sessionFactory.getCurrentSession()。当时遇到了相同的错误。
user811433 '16

2
这给出了一种非常令人讨厌的行为,即如果sessionFactory.getCurrentSession();成功,则不应关闭该会话,但sessionFactory.openSession();必须关闭该会话
Richard Tingle

1
同意@RichardTingle。似乎openSession是绕过异常的黑客。空闲的解决方案应该是什么?
Praveen Shendge '18 -10-11

@Praveen我实际上所做的是让一个服务接受一个lambda,Function<Session,T>意思是“如果我有一个会话,我会用它来做X”。然后,该方法处理预配和(如果需要)取消配置会话,并只返回T。因此,服务的外部使用者实际上从未参与过会话
Richard Tingle

这使我的程序运行无错误迹象。我担心我没有跨这种方式创建的多个查询之间的事务,这意味着我可能会返回不一致的结果吗?
Ole VV


3

在您的xyz.DAOImpl.java中

请执行以下步骤:

//步骤1:设置会话工厂

@Resource(name="sessionFactory")
private SessionFactory sessionFactory;

public void setSessionFactory(SessionFactory sf)
{
    this.sessionFactory = sf;
}

//步骤2:尝试获取当前会话,并捕获HibernateException异常。


//第3步:如果有任何HibernateException异常,则为true以获取openSession。

try 
{
    //Step-2: Implementation
    session = sessionFactory.getCurrentSession();
} 
catch (HibernateException e) 
{
    //Step-3: Implementation
    session = sessionFactory.openSession();
}

嗨!休眠不应该自己做吗?
克里斯(Chris

2

我在web.xml中添加了这些配置,对我来说效果很好!

<filter>
    <filter-name>OpenSessionInViewFilter</filter-name>
    <filter-class>org.springframework.orm.hibernate5.support.OpenSessionInViewFilter</filter-class>
    <init-param>
        <param-name>sessionFactoryBeanName</param-name>
        <param-value>sessionFactory</param-value>
    </init-param>
    <init-param>
        <param-name>flushMode</param-name>
        <param-value>AUTO</param-value>
    </init-param>
</filter>
<filter-mapping>
    <filter-name>OpenSessionInViewFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

此外,排名最高的答案为我提供了一些线索,以防止应用程序在第一次运行时出现恐慌。


1
我正在使用springMVC 4和Hibernate 5
何德福

2

您需要允许交易到您的DAO方法。加,

@Transactional(readOnly = true, propagation=Propagation.NOT_SUPPORTED)

道方法 并且@Transactional应该来自包装:

org.springframework.transaction.annotation.Transactional

1

我也有此错误,因为在使用@Transactional 批注的文件中,我导入了错误的类

import javax.transaction.Transactional; 

代替javax,使用

import org.springframework.transaction.annotation.Transactional; 

1

我的解决方案是(使用Spring)将失败的方法放入另一个创建并提交事务的方法中。

为此,我首先注入了以下内容:

@Autowired
private PlatformTransactionManager transactionManager;

最后做到了:

public void newMethod() {
    DefaultTransactionDefinition definition = new DefaultTransactionDefinition();
    TransactionStatus transaction = transactionManager.getTransaction(definition);

    oldMethod();

    transactionManager.commit(transaction);
}

1

@Transactional =javax.transaction.Transactional。放在旁边@Repository


0

我的配置是这样的。我有一个QuartzJob,一个Service Bean和Dao。像往常一样,它是使用LocalSessionFactoryBean(用于hibernate)和SchedulerFactoryBean用于Quartz框架配置的。而写石英工作,我误用@注解它服务,我不应该这样做,因为我是用另一种策略,接线QuartzBean使用AutowiringSpringBeanJobFactory延伸SpringBeanJobFactory

所以,实际上正在发生的是,由于石英自动装配,TX渐渐注入工作豆,并在同一时间Tx背景是凭借@设置服务注解,因此TX正在下降同步进行!

我希望它对上述解决方案确实无法解决问题的用户有所帮助。我正在使用Spring 4.2.5和Hibernate 4.0.1,

我看到在此线程中,有不必要的建议将@ Transactional批注添加到DAO(@ Repository),这是一个无用的建议,因为@ Repository具有所有需要的内容,而不必专门设置@ transactional为DAO,就像从@Trasancational注入的服务中调用DAO 一样。我希望这对一起使用Quartz,Spring和Hibernate的人们可能有所帮助。


0

添加transaction-manager到您<annotation-driven/>spring-servlet.xml中:

<tx:annotation-driven transaction-manager="yourTransactionBeanID"/>

0

检查您的dao班。必须是这样的:

Session session = getCurrentSession();
Query query = session.createQuery(GET_ALL);

和注释:

@Transactional
@Repository

0

我遇到了同样的问题,最终发现<tx:annotaion-driven />[dispatcher]-servlet.xmlwhere component-scan元素启用带@service注释的类中未定义。

只需将<tx:annotaion-driven />组件扫描元素放在一起,问题就消失了。


0

我的类似问题通过以下两种方法得以解决。

1)通过手动处理交易:

Session session = sessionFactory.getCurrentSession();
Transaction tx = session.beginTransaction();
UserInfo user = (UserInfo) session.get(UserInfo.class, 1);
tx.commit();

2)告诉Spring在您的web.xml过滤器中为您打开和管理交易,并确保使用@Repository @Transactional

<filter>
  <filter-name>hibernateFilter</filter-name>
  <filter-class>org.springframework.orm.hibernate5.support.OpenSessionInViewFilter</filter-class>
  <init-param>
    <param-name>sessionFactory</param-name>
    <param-value>session.factory</param-value>
  </init-param>
</filter>
<filter-mapping>
  <filter-name>hibernateFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>

0

感谢您对mannedear的评论。我使用springmvc,在我的情况下,我必须使用

@Repository
@Transactional
@EnableTransactionManagement
public class UserDao {
...
}

并且我还将spring-context添加到pom.xml中,它可以工作


0

我遇到过同样的问题。我解决了以下问题:

  1. 将此行添加到dispatcher-servlet文件中:

    <tx:annotation-driven/>

    <beans>在同一文件中检查以上部分。必须存在以下两行:

    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation= "http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd"
  2. 另外,请确保已添加@Repository以及@Transactional使用位置sessionFactory

    @Repository
    @Transactional
    public class ItemDaoImpl implements ItemDao {
        @Autowired
        private SessionFactory sessionFactory;

-1

在上面的此类中,@Repository只需再放置一个注释即可@Transactional。如果可行,请回覆(Y/ N):

@Repository
@Transactional
public class StudentDAOImpl implements StudentDAO

1
欢迎来到SO。请仔细阅读如何写一个好的答案。在SO中,没有回答Y / N的习惯。如果您的答案适用于该人,他们将标记为已接受。一个有用的答案可能也被推荐。
Sri9911
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.