解决Spring Data JPA延迟加载could not initialize proxy - no Session 错误

项目采用spring boot构建,提供restful接口给PHP调用,接口类统一使用@RestController注解

 

实体关系

 

[java] view plain copy
 
  1. @ManyToOne(fetch = FetchType.LAZY)  
[java] view plain copy
 
  1. @JoinColumn(name = "department_id")  
  2. private Department department;  

 

 

 

[java] view plain copy
 
  1. @OneToMany(mappedBy = "department", fetch = FetchType.LAZY)  
  2. private Set<Employee> employees;  



 

单元测试

[java] view plain copy
 
  1. @Test  
  2. public void testGet() {  
  3.     Department dept = deptService.get(1L);  
  4.     Assert.assertEquals("IT", dept.getName());  
  5.     Set<Employee> employees = dept.getEmployees();  
  6.     Assert.assertEquals(2, employees.size());  
  7. }  

 

 

异常信息

 

[java] view plain copy
 
  1. org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.ai.demo.entity.Department.employees, could not initialize proxy - no Session  
  2.     at org.hibernate.collection.internal.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:587)  
  3.     at org.hibernate.collection.internal.AbstractPersistentCollection.withTemporarySessionIfNeeded(AbstractPersistentCollection.java:204)  
  4.     at org.hibernate.collection.internal.AbstractPersistentCollection.readSize(AbstractPersistentCollection.java:148)  
  5.     at org.hibernate.collection.internal.PersistentSet.size(PersistentSet.java:143)  
  6.     at com.ai.demo.service.DepartmentServiceTests.testGet(DepartmentServiceTests.java:37)  
  7.     at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)  
  8.     at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)  
  9.     at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)  
  10.     at java.lang.reflect.Method.invoke(Method.java:498)  
  11.     at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)  
  12.     at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)  
  13.     at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)  
  14.     at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)  
  15.     at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:75)  
  16.     at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)  
  17.     at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)  
  18.     at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)  
  19.     at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:252)  
  20.     at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:94)  
  21.     at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)  
  22.     at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)  
  23.     at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)  
  24.     at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)  
  25.     at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)  
  26.     at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)  
  27.     at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)  
  28.     at org.junit.runners.ParentRunner.run(ParentRunner.java:363)  
  29.     at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:191)  
  30.     at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86)  
  31.     at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)  
  32.     at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)  
  33.     at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:678)  
  34.     at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)  
  35.     at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)  

 

解决LazyInitializationException异常大概有这么几种方式
1.关闭LazyInitialization, 将fetch设成eager

2.在spring boot的配置文件application.properties添加spring.jpa.open-in-view=true

3.用spring 的OpenSessionInViewFilter

第一种方式显然不好,无法使用到延迟加载的特性,会带来性能问题

后面两种方式只能用在Servlet容器下,而当我们在spring boot环境下运行单元测试的时候是无法启用OpenSessionInViewFilter的

 

其实要解决这种情况下的问题也很简单,只需要在单元测试方法@Transactional注解即可解决

 

[java] view plain copy
 
    1. @Test  
    2. @Transactional  
    3. public void testGet() {  
    4.     Department dept = deptService.get(1L);  
    5.     Assert.assertEquals("IT", dept.getName());  
    6.     Set<Employee> employees = dept.getEmployees();  
    7.     Assert.assertEquals(2, employees.size());  
    8. }  

转载至:https://blog.csdn.net/chrislyl/article/details/54630413

posted @ 2018-04-27 16:02  猴子1  阅读(2332)  评论(0编辑  收藏  举报