解決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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM