Spring Hibernate - ไม่สามารถรับ Session ที่ซิงโครไนซ์ธุรกรรมสำหรับเธรดปัจจุบัน


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)
);

3
คุณลองเพิ่ม @Transactional ในวิธีการสร้าง DAO ของคุณหรือไม่?
John

1
คุณลืมประกาศ HibernateTransactionManager และสร้างวิธีการโดยใช้ Hibernate transactional
JB Nizet

@itachi ไม่ถูกต้องsessionFactory.openSession()ธุรกรรมจะปิดการใช้งาน เพราะไม่ใช่เซสชันเดียวกัน > เพิ่มคำอธิบายประกอบ @Transactional of spring ในบริการคลาส @Patrikoko ถูกต้อง! ดูคำถามนี้: stackoverflow.com/questions/15620355/…ตัวอย่าง:@Transactional(readOnly = true, propagation = Propagation.REQUIRED, rollbackFor = {java.lang.Exception.class})
nvnpnco

คำตอบ:


202

คุณต้องเปิดการใช้งานการสนับสนุนการทำธุรกรรม ( <tx:annotation-driven>หรือ@EnableTransactionManagement) และประกาศและมันควรจะทำงานผ่านtransactionManagerSessionFactory

คุณต้องเพิ่ม@Transactionalลงในไฟล์@Repository

ด้วย@Transactionalใน@RepositorySpring ของคุณสามารถใช้การสนับสนุนการทำธุรกรรมกับที่เก็บของคุณ

Studentชั้นเรียนของคุณไม่มีคำอธิบายประกอบ @ javax.persistence * อย่างไร@Entityฉันสมมติว่าการกำหนดค่าการแมปสำหรับคลาสนั้นได้รับการกำหนดผ่าน XML


1
ได้โปรดเขียนโค้ดของแอปพลิเคชันได้ไหมเพราะมันใช้งานไม่ได้ .. นี่เป็นแอปพลิเคชั่นแรกของฉันกับ Hibernate
Alex

3
คำอธิบายประกอบที่เทียบเท่ากับ <tx: annotation-driven> คือ @EnableTransactionManagement
Anand Rockzz

6
ตรวจสอบให้แน่ใจว่าคุณใช้
org.springframework.transaction.annotation.Transactional

ไชโยเพื่อนฉันไม่อยากจะเชื่อเลยว่าฉันพลาดคำอธิบายประกอบนี้ :)
Boldbayar

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();สำเร็จจะต้องปิด
Richard Tingle

1
เห็นด้วย @RichardTingle ดูเหมือนว่า openSession เป็นการแฮ็กเพื่อข้ามข้อยกเว้น ทางออกที่ไม่ได้ใช้งานสำหรับสิ่งนี้ควรเป็นอย่างไร?
Praveen Shendge

@ Praveen สิ่งที่ฉันทำจริงๆคือมีบริการยอมรับแลมด้า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 ดังนั้นจึงเป็นจริงเพื่อรับ openSession

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

ไฮ! ไฮเบอร์เนตไม่ควรทำด้วยตัวเอง?
คริส

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)

มากกว่าวิธีการ dao ของคุณ และ@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);
}


0

การกำหนดค่าของฉันเป็นแบบนี้ ฉันมีQuartzJob , Service Bean และ Dao ตามปกติจะถูกกำหนดค่าด้วย LocalSessionFactoryBean (สำหรับ hibernate) และ SchedulerFactoryBean สำหรับเฟรมเวิร์ก Quartz ในขณะที่เขียนงานควอตซ์ผมโดยไม่ได้ตั้งใจข้อเขียนด้วย @ บริการผมไม่ควรทำอย่างนั้นเพราะผมใช้กลยุทธ์อื่นเพื่อลวดQuartzBeanใช้AutowiringSpringBeanJobFactoryขยายSpringBeanJobFactory

สิ่งที่เกิดขึ้นจริงก็คือเนื่องจาก Quartz Autowire ทำให้ TX ถูกฉีดไปที่ Job Bean และในเวลาเดียวกัน Tx Context ก็ถูกตั้งค่าโดยอาศัยคำอธิบายประกอบ@ Serviceและด้วยเหตุนี้ TX จึงไม่ตรงกัน !!

ฉันหวังว่ามันจะช่วยสำหรับผู้ที่วิธีแก้ปัญหาข้างต้นไม่สามารถแก้ปัญหาได้ ฉันใช้ Spring 4.2.5 และ Hibernate 4.0.1

ผมเห็นว่าในหัวข้อนี้จะมีข้อเสนอแนะที่ไม่จำเป็นเพื่อเพิ่ม @ Transactionalคำอธิบายประกอบถึง DAO นี้ (@ Repository ) ที่เป็นสาเหตุข้อเสนอแนะที่ไร้ประโยชน์ @ Repositoryมีทุกสิ่งที่มันต้องมีไม่ได้มีการตั้งเป็นพิเศษที่ @ การทำธุรกรรมบน DAOs เป็น DAOs จะเรียกว่าจากการให้บริการที่มีอยู่แล้วจะถูกฉีดโดย@Trasancational ฉันหวังว่านี่อาจเป็นประโยชน์สำหรับผู้ที่ใช้ Quartz, Spring และ Hibernate ร่วมกัน


0

เพิ่มtransaction-managerที่คุณ<annotation-driven/>ในฤดูใบไม้ผลิ 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.xmlสแกนองค์ประกอบที่เปิดใช้งาน@service

เพียงใส่<tx:annotaion-driven />องค์ประกอบสแกนองค์ประกอบเข้าด้วยกันปัญหาก็หายไป


0

ปัญหาที่คล้ายกันของฉันได้รับการแก้ไขด้วย 2 วิธีด้านล่าง

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 {
...
}

และฉันยังเพิ่มบริบทฤดูใบไม้ผลิใน 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 กรุณาอ่านฉันจะเขียนคำตอบที่ดีได้อย่างไร ดังนั้นจึงไม่มีธรรมเนียมในการตอบกลับ Y / N หากคำตอบของคุณเหมาะกับบุคคลที่พวกเขาจะทำเครื่องหมายว่าเป็นที่ยอมรับ คำตอบที่เป็นประโยชน์อาจได้รับการโหวตด้วย
Sri9911
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.