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

106

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

student.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)
);
Alex
источник
3
Вы пробовали добавить @Transactional в свой метод создания DAO?
Джон
1
Вы забыли объявить HibernateTransactionManager и сделать метод, использующий Hibernate, транзакционным.
JB Nizet
@itachi неверен, sessionFactory.openSession()транзакция отключится. Потому что это разные сеансы. > Добавить аннотацию @Transactional of spring в класс service @Patrikoko правильно! см. этот вопрос: stackoverflow.com/questions/15620355/… пример:@Transactional(readOnly = true, propagation = Propagation.REQUIRED, rollbackFor = {java.lang.Exception.class})
nvnpnco

Ответы:

201

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

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

С @Transactionalвашей @RepositorySpring может применять поддержку транзакций в репозиторий.

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

Мануэль Джордан
источник
1
Пожалуйста, не могли бы вы написать код приложения, потому что оно не работает .. Это мое первое приложение с Hibernate
Alex
3
Аннотационный эквивалент <tx: annotation-driven> - @EnableTransactionManagement
Ананд Рокзз
6
Также убедитесь, что вы используете org.springframework.transaction.annotation.Transactional, а не javax.persistance.Transactional
imnd_neel
Ура, приятель, не могу поверить, что пропустил эту аннотацию :).
Boldbayar 01
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
2
Это дает действительно неприятное поведение: в случае sessionFactory.getCurrentSession();успеха сеанс не должен закрываться, но в случае sessionFactory.openSession();успеха он должен быть закрыт
Ричард Тингл
1
Согласен @RichardTingle. Похоже, что openSession - это способ обойти исключение. Каким должно быть для этого праздное решение?
Правин Шендж
@Praveen на самом деле я заставил службу принять лямбду, Function<Session,T>означающую «если бы у меня был сеанс, я бы использовал его для выполнения X». Затем метод обрабатывает подготовку и (при необходимости) деинициализацию сеанса и просто возвращает T. Таким образом, внешние потребители службы фактически никогда не получат доступ к сеансу
Ричард Тингл,
Это заставило мою программу работать без признаков ошибки. Меня беспокоит, что у меня нет транзакции для нескольких запросов, созданных таким образом, а это означает, что я рискую вернуть непоследовательные результаты?
Оле В.В.,
13

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

Патрико
источник
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();
}
АрунДхвадж Иитх
источник
Здравствуй! Разве Hibernate не должен делать это сам по себе?
Крис
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 (для гибернации) и SchedulerFactoryBean для структуры Quartz. при написании задания Quartz я по ошибке аннотировал его с помощью @ Service , я не должен был этого делать, потому что я использовал другую стратегию для подключения QuartzBean с помощью AutowiringSpringBeanJobFactory, расширяющего SpringBeanJobFactory .

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

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

Я вижу, что в этом потоке есть ненужное предложение добавить аннотацию @ Transactional к DAO (@ Repository ), это бесполезное предложение, потому что @ Repository имеет все, что ему нужно, не нужно специально устанавливать этот @ transactional на DAO, так как DAO вызываются из служб, которые уже были введены @Trasancational . Надеюсь, это может быть полезно людям, которые вместе используют Quartz, Spring и Hibernate.

САНДЖАЙ ГАУТАМ
источник
0

Добавить transaction-managerв свой <annotation-driven/>в весенне-servlet.xml:

<tx:annotation-driven transaction-manager="yourTransactionBeanID"/>
Маджид
источник
0

Проверьте свой класс дао. Должно быть так:

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

И аннотации:

@Transactional
@Repository
Евгения О
источник
0

Я столкнулся с той же проблемой и, наконец, обнаружил, что <tx:annotaion-driven />не был определен в [dispatcher]-servlet.xmlэлементе where 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 нет обычая отвечать Да / Нет. Если ваш ответ работает для человека, которого они отметят, он считается принятым. Также можно проголосовать за полезный ответ.
Sri9911,