没有可用的事务性 EntityManager - 使用 JPA Api,Hibernate 会话出错
Posted
技术标签:
【中文标题】没有可用的事务性 EntityManager - 使用 JPA Api,Hibernate 会话出错【英文标题】:No transactional EntityManager available - working with JPA Api, error with Hibernate Session 【发布时间】:2014-04-18 10:37:33 【问题描述】:我正在尝试从注入的 EntityManager 中解开 Hibernate Session,因为我需要使用 Hibernate 的本机 Criteria API。
当我尝试使用 Criteria 时,出现以下异常:
Caused by: java.lang.IllegalStateException: No transactional EntityManager available
at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:224)
at com.sun.proxy.$Proxy28.unwrap(Unknown Source)
at sk.uniba.ais2.fajr.dao.impl.PouzivatelDAOHibernate.findByLogin(PouzivatelDAOHibernate.java:22)
at sk.uniba.ais2.fajr.bo.PouzivatelService.findByLogin(PouzivatelService.java:20)
at sk.uniba.ais2.fajr.mvc.controller.FooController.getFoooFOO(FooController.java:38)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
at org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:215)
at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:132)
at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:104)
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandleMethod(RequestMappingHandlerAdapter.java:749)
at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:690)
at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:83)
at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:945)
at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:876)
at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:961)
... 62 more
但是,当我使用 EntityManager 本身的 Query api 时,一切正常。
我的配置:
db.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:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
default-autowire="byName" xmlns:jpa="http://www.springframework.org/schema/data/jpa"
xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/data/jpa http://www.springframework.org/schema/data/jpa/spring-jpa-1.3.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">
<!-- Scans within the base package of the application for @Components to
configure as beans -->
<bean id="placeholderConfig"
class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="location" value="classpath:db.properties" />
</bean>
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
destroy-method="close">
<property name="driverClass" value="$db.driver" />
<property name="jdbcUrl" value="$db.url" />
<property name="user" value="$db.username" />
<property name="password" value="$db.password" />
</bean>
<bean id="jpaVendorAdapter"
class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="showSql" value="true" />
<property name="database" value="ORACLE" />
</bean>
<bean id="entityManagerFactory"
class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter" ref="jpaVendorAdapter" />
<property name="packagesToScan" value="sk.uniba.ais2.fajr.entities" />
<property name="jpaPropertyMap">
<map>
<entry key="hibernate.default_schema" value="AIS2" />
</map>
</property>
</bean>
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<tx:annotation-driven transaction-manager="transactionManager"
proxy-target-class="true" />
<jpa:repositories base-package="sk.uniba.ais2.fajr.dao" />
</beans>
PouzivatelDAOHibernate 不起作用:
@Repository
public class PouzivatelDAOHibernate implements PouzivatelDAO
@PersistenceContext
private EntityManager entityManager;
@Override
public Pouzivatel findByLogin(String login)
Criteria criteria = entityManager.unwrap(Session.class).createCriteria(Pouzivatel.class);
return (Pouzivatel) criteria.add(Restrictions.eq("login", login)).uniqueResult();
PouzivatelDAOHibernate 可以工作:
@Repository
public class PouzivatelDAOHibernate implements PouzivatelDAO
@PersistenceContext
private EntityManager entityManager;
@Override
public Pouzivatel findByLogin(String login)
// Criteria criteria = entityManager.unwrap(Session.class).createCriteria(Pouzivatel.class);
// return (Pouzivatel) criteria.add(Restrictions.eq("login", login)).uniqueResult();
Query query = entityManager.createQuery("select p from Pouzivatel p where login=?");
query.setParameter(1, login);
return (Pouzivatel) query.getSingleResult();
编辑:我正在使用 Spring Data JPA
【问题讨论】:
【参考方案1】:我实际上已经解决了这个问题(解决方案非常简单):使用org.springframework.transaction.annotation.Transactional
而不是javax.transaction.Transactional
【讨论】:
【参考方案2】:考虑在您的 DAO 方法上使用 @Transactional
。看来你的配置是正确的。
【讨论】:
我也试过了,还是没有 不 :) ***.com/questions/1079114/… 这个特定应用程序的目的是仅从数据库加载实体,进行一些后处理(将实体映射到一些 DTO)并通过一些 REST 端点发布数据。完全只读,没有业务逻辑,这就是我在这种情况下完全跳过服务层的原因。【参考方案3】:通过添加单个 bean 定义,Spring 容器将充当 JPA 容器并从您的 EntityManagerFactory 注入一个 EnitityManager。
看 https://spring.io/blog/2006/08/07/using-jpa-in-spring-without-referencing-spring
【讨论】:
不幸的是,这不是这个问题的答案——我已经让 PersistenceContext 工作了,唯一的问题是事务处理以上是关于没有可用的事务性 EntityManager - 使用 JPA Api,Hibernate 会话出错的主要内容,如果未能解决你的问题,请参考以下文章
javax.persistence.TransactionRequiredException:没有可用的事务性 EntityManager
java.lang.IllegalStateException:没有可用的事务性 EntityManager
Spring boot + Hibernate + JPA 没有可用的事务性 EntityManager
TransactionRequiredException:没有用于当前线程的具有实际事务的EntityManager-无法可靠地处理“合并”调用
javax.persistence.TransactionRequiredException:没有 EntityManager 具有可用于当前线程的实际事务