Spring Hibernate - не удалось получить сеанс с синхронизацией транзакций для текущего потока
Я создал приложение с spring + hibernate, но всегда получаю эту ошибку. Это мое первое приложение с спящим режимом, я прочитал несколько руководств, но я не могу решить эту проблему. Где я делаю неправильно?
Это код моего приложения
ott 05, 2014 4:03:06 PM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
Informazioni: Refreshing org[email protected]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)
);
Ответы
Ответ 1
Вы должны включить поддержку транзакций (<tx:annotation-driven>
или @EnableTransactionManagement
) и объявить transactionManager
, и она должна работать через SessionFactory
.. p >
Вы должны добавить @Transactional
в свой @Repository
С @Transactional
в вашем @Repository
Spring можно применить транзакционную поддержку в вашем репозитории.
В вашем классе Student
нет @@javax.persistence. * аннотаций, как @Entity
, я предполагаю, что конфигурация сопоставления для этого класса определена через XML.
Ответ 2
У меня была такая же проблема, но в классе, который не был частью уровня сервиса. В моем случае менеджер транзакций был просто получен из контекста с помощью метода getBean()
, а класс принадлежал к слою представления - в моем проекте используется метод OpenSessionInView
.
Метод sessionFactory.getCurrentSession()
вызывает то же исключение, что и авторское. Решение для меня было довольно простым.
Session session;
try {
session = sessionFactory.getCurrentSession();
} catch (HibernateException e) {
session = sessionFactory.openSession();
}
Если метод getCurrentSession()
выходит из строя, openSession()
должен выполнить трюк.
Ответ 3
Добавить аннотацию @Transactional spring в службе класса
Ответ 4
В xyz.DAOImpl.java
Выполните следующие действия:
//Шаг-1: Установить сеанс 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();
}
Ответ 5
Я добавил эту конфигурацию в 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>
Кроме того, наиболее ранжированный ответ дает мне подсказку, чтобы предотвратить панику при первом запуске.
Ответ 6
@Transactional = javax.transaction.Transactional помещает его рядом с @Repository
Ответ 7
У меня тоже была эта ошибка, потому что в файле, где я использовал аннотацию @Transactional
, я импортировал неправильный класс
import javax.transaction.Transactional;
Вместо javax используйте
import org.springframework.transaction.annotation.Transactional;
Ответ 8
Вам необходимо разрешить транзакцию для вашего метода DAO.
Добавить,
@Transactional(readOnly = true, propagation=Propagation.NOT_SUPPORTED)
по вашим методам дао.
И @Transactional
должен быть из пакета:
org.springframework.transaction.annotation.Transactional
Ответ 9
Мое решение было (с помощью Spring) поместить метод, который завершается с ошибкой, в другой метод, который создает и фиксирует транзакцию.
Для этого я сначала ввел следующее:
@Autowired
private PlatformTransactionManager transactionManager;
И наконец сделал это:
public void newMethod() {
DefaultTransactionDefinition definition = new DefaultTransactionDefinition();
TransactionStatus transaction = transactionManager.getTransaction(definition);
oldMethod();
transactionManager.commit(transaction);
}
Ответ 10
@itachi
неверно, sessionFactory.openSession()
транзакция будет отключена. Потому что это не тот же сеанс.
Добавить аннотацию @Transactional spring в службе класса
@Патрикоко верен!
см. этот вопрос: Для веб-приложения MVC spring должно ли @Transactional перейти на контроллер или услугу?
Пример:
@Transactional(readOnly = true, propagation = Propagation.REQUIRED, rollbackFor = {java.lang.Exception.class})
Ответ 11
Я столкнулся с той же проблемой и, наконец, узнал, что он не был определен в [диспетчере] -servlet.xml, где элемент компонента-компонента включен в аннотированный класс @service.
Проще говоря, вместе с элементом компонентного сканирования проблема исчезла.
Ответ 12
Моя аналогичная проблема была исправлена с менее чем двумя подходами.
-
Выполняя транзакции вручную.
Session session = sessionFactory.getCurrentSession();
Транзакция tx = session.beginTransaction();
UserInfo user = (UserInfo) session.get(UserInfo.class, 1);
tx.commit();
-
Сообщите Spring, чтобы открыть и управлять транзакциями для вас в ваших фильтрах web.xml и обеспечить использование @Repository @Transactional
< фильтp > < filter-name > hibernateFilter </имя-фильтрa > < filter-class > org.springframework.orm.hibernate5.support.OpenSessionInViewFilter </filter-class > < init-param > < Имя PARAM > SessionFactory < пары-значение > session.factory </init-param > </фильтp > < фильтрация карт > < -Фильтp > hibernateFilter </имя-фильтрa > < URL-шаблон > /* </url-pattern > </filter-mapping >
Извините за неправильное форматирование.
Ответ 13
Моя конфигурация была такой. У меня был QuartzJob, сервис Bean и дао. как обычно, он был настроен с помощью LocalSessionFactoryBean (для спящего режима) и SchedulerFactoryBean для каркаса. при написании задания Quartz я по ошибке аннотировал его с помощью @ Service, я не должен был этого делать, потому что использовал другую стратегию для подключения QuartzBean, используя AutowiringSpringBeanJobFactory, расширяющий SpringBeanJobFactory.
Итак, что на самом деле происходило, так это то, что из-за Quartz Autowire TX был введен в Job Bean, и в то же время Tx Context был установлен в силу @ Service аннотации и, следовательно, TX падала из синхронизации!
Я надеюсь, что это поможет тем, для кого выше решения действительно не решили проблему. Я использовал Spring 4.2.5 и Hibernate 4.0.1,
Я вижу, что в этом потоке есть ненужное предложение добавить @ Transactional аннотацию в DAO (@ Репозиторий), что является бесполезной причиной предложения @ Репозиторий имеет все, что ему не нужно, чтобы специально установить этот @ транзакционный в DAO, поскольку DAO вызывается из служб, которые уже были введены с помощью @Trasancational. Надеюсь, это могут быть полезные люди, которые используют Quartz, Spring и Hibernate вместе.
Ответ 14
Добавьте transaction-manager
в ваш <annotation-driven/>
файл spring-servlet.xml:
<tx:annotation-driven transaction-manager="yourTransactionBeanID"/>
Ответ 15
Проверьте свой дао класс. Должно быть так:
Session session = getCurrentSession();
Query query = session.createQuery(GET_ALL);
И аннотации:
@Transactional
@Repository
Ответ 16
В этом классе выше @Repository только что поместили еще одну аннотацию @Transactional, она будет работать. Если это работает, ответить назад (Y/N)
@Repository
@Transactional
Открытый класс StudentDAOImpl реализует StudentDAO