java - Spring 4 Hibernate 4 JPA LazyInitializationException:无法延迟初始化角色集合

原文 标签 java spring hibernate jpa lazy-initialization

我与Spring 4和Hibernate 4和JPA一起使用建立了数据库连接。当我尝试获取懒惰的初始化字段时,出现LazyInitializationException。

我的applicationContext.xml:

<context:annotation-config />
<context:component-scan base-package="xy" />

<jpa:repositories base-package="xy.repositories" />

<bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="persistenceUnitName" value="punit" />
    <property name="dataSource" ref="dataSource" />

    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="showSql" value="true" />
        </bean>
    </property>
    <property name="jpaPropertyMap">
        <map>
            <entry key="hibernate.dialect" value="org.hibernate.dialect.MySQL5InnoDBDialect" />
            <entry key="hibernate.hbm2ddl.auto" value="update" />
            <entry key="hibernate.format_sql" value="true" />
        </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" />

<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
    <property name="driverClassName" value="com.mysql.jdbc.Driver" />
    <property name="url" value="jdbc:mysql://localhost:3306/keysystem" />
    <property name="username" value="username" />
    <property name="password" value="password" />
</bean>

<bean class="org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor"/>


模型:

@Entity
@Table
public class Employee {
    @Id
    @GeneratedValue
    private long employeeId;
    private String firstName;
    private String lastName;
    @OneToMany(fetch = FetchType.LAZY, mappedBy = "employee")
    @Cascade(CascadeType.ALL)
    @Fetch(FetchMode.SELECT)
    private Set<Legitimacy> legitimacies;
    ...


我只找到了MVC过滤器解决方案,但没有使用MVC。我尝试了Hibernate.initialize(...)和@ Transational,@ Fetch批注,但是这些并没有解决我的问题。

这是堆栈跟踪:

Caused by: org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: xy.Employee.legitimacies, could not initialize proxy - no Session
    at org.hibernate.collection.internal.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:575)
    at org.hibernate.collection.internal.AbstractPersistentCollection.withTemporarySessionIfNeeded(AbstractPersistentCollection.java:214)
    at org.hibernate.collection.internal.AbstractPersistentCollection.initialize(AbstractPersistentCollection.java:554)
    at org.hibernate.collection.internal.AbstractPersistentCollection.read(AbstractPersistentCollection.java:142)
    at org.hibernate.collection.internal.PersistentSet.iterator(PersistentSet.java:180)
    ... 64 more

最佳答案

Guillaume是正确的,您在代码中某个位于事务上下文之外的位置中使用了Hibernate管理的代理。

因此,当您调用尚未初始化的getLegitimacies()时,Hibernate尝试通过抓取会话并进行数据库调用来填充其代理。

例如,如果您在控制器中并且没有会话可用,则会引发异常。您必须在dao的某个位置获取合法性设置(例如,通过在HQL查询中使用FETCH关键字):

FROM Employee e LEFT JOIN FETCH e.legitimacies WHERE <your_predicate>;


如果不在服务/传输层中,则也不应操作实体,而应使用DTO以避免这些问题。

相关文章:

java - 如果从IntelliJ IDEA运行应用程序,则找不到Mapstruct生成的类

java - 如何扫描Hibernate实体的包而不是使用hbm.xml?

java - 如何解决java.lang.NoClassDefFoundError:org / hibernate / SessionFactory?

java - 休眠:如何调用返回varchar的存储的函数?

java - log4j,外部日志记录

java - 通过反射获取字段名称是否应该避免进行昂贵的操作?

java - Log4J2不输出调试信息

java - 在model.addAttribute(“ name”,value)和mv.addObject(“ name”,value)之间有区别吗?

java - 春季行动下载不适用于Android浏览器

java - 将DTO的接口传递给DAO可以吗