Spring Hibernate - не удалось получить транзакцию-синхронизированный сеанс для текущего потока


Я создал приложение с 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

studentDAO.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)
);
13 78

13 ответов:

вы должны включить поддержка транзакций (<tx:annotation-driven> или @EnableTransactionManagement) и объявить the transactionManager и он должен работать через SessionFactory.

вы должны добавить @Transactional в своем @Repository

С @Transactional в своем @Repository Spring может применять транзакционную поддержку в вашем репозитории.

код Student класс не имеет @javax.стойкость.* аннотации как @Entity, Я предполагаю, что конфигурация отображения для этот класс был определен через XML.

у меня была такая же проблема, но в классе, который не был частью слоя сервиса. В моем случае менеджер транзакций был просто получен из контекста с помощью getBean() метод, и класс принадлежал к слою представления-мой проект использует OpenSessionInView техника.

The sessionFactory.getCurrentSession() метод, вызывает такое же исключение, как автора. Решение для меня было довольно просто.

Session session;

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

если getCurrentSession() метод терпит неудачу,openSession() должны сделать уловка.

добавьте аннотацию @Transactional spring в класс service

@Transactional =javax.торговая операция.Транзакционный положите его рядом с @Repository

в вашем xyz.Даоимпл.java

выполните следующие действия:

/ / Step-1: Set session factory

@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();
}

Я добавил эти настройки в 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>

кроме того, самый ранжированный ответ дает мне подсказки, чтобы предотвратить панику приложения при первом запуске.

у меня тоже эта ошибка, потому что в файле, где я использовал @Transactional аннотация, я импортировал не тот класс

import javax.transaction.Transactional; 

вместо javax, используйте

import org.springframework.transaction.annotation.Transactional; 

Я столкнулся с той же проблемой и, наконец, узнал, что не был определен в [диспетчер]-сервлет.XML, в котором компонент-сканирующий элемент включен @обслуживание аннотированного класса.

проще говоря с элементом component-scan вместе, проблема исчезла.

моя аналогичная проблема была исправлена с помощью ниже 2 подходов.

  1. через ручную обработку транзакций.

    Session session = sessionFactory.getCurrentSession();

    транзакция tx = сессия.beginTransaction();

    userInfo user = (UserInfo) сеанс.Вам(пользователи.класс, 1);

    tx.commit ();

  2. скажите Spring, чтобы открыть и управлять транзакциями для вас в вашем интернете.xml фильтрует и обеспечивает чтобы использовать @Repository @Transactional

    hibernateFilter org.springframework.ОЗР.hibernate5.поддержка.OpenSessionInViewFilter sessionFactory сеанс .фабрика hibernateFilter /*

извините за неправильное форматирование.

моя конфигурация была такой. У меня был QuartzJob, служебный компонент и Dao . как обычно, он был настроен с LocalSessionFactoryBean (для hibernate) и SchedulerFactoryBean для Quartz framework. при написании задания Quartz я по ошибке аннотировал его с помощью @сервис , Я не должен был этого делать, потому что я использую другую стратегию для провода QuartzBean используя AutowiringSpringBeanJobFactory расширения SpringBeanJobFactory.

Итак, что на самом деле происходило , так это то, что из-за Quartz Autowire TX вводился в Боб задания, и в то же время контекст Tx был установлен в силу @сервис аннотация и, следовательно, TX выпадал из синхронизации !!

Я надеюсь, что это поможет тем, для кого выше решения действительно не решили проблему. Я использовал Spring 4.2.5 и Hibernate 4.0.1,

Я вижу, что в этой теме есть ненужное предложение добавить @транзакционные аннотация к DAO (@хранилище), это бесполезное предложение причина @хранилище имеет все, что ему нужно, не нужно специально устанавливать это @транзакционные на DAOs, так как DAOs вызываются из служб, которые уже вводятся @Trasancational . Я надеюсь , что это может быть полезно людям, которые используют кварц, весну и спящий режим вместе.

добавить transaction-manager на <annotation-driven/> на spring-сервлет.XML-код:

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

проверьте свой класс dao. Это должно быть так:

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

и аннотации:

@Transactional
@Repository

вам нужно разрешить транзакцию для вашего метода DAO. Добавь,

@Transactional(readOnly = true, propagation=распространение.NOT_SUPPORTED)

над вашими методами dao. И @Transactional shoul быть пакета,

орг.springframework.торговая операция.аннотация.Транзакционный