您好,登錄后才能下訂單哦!
這篇文章主要講解了“Hibernate Lazy加載問題怎么解決”,文中的講解內(nèi)容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Hibernate Lazy加載問題怎么解決”吧!
Hbm文件
<bean id="BookItemRepositoryImpl"
class="domain.repository.hibernate.BookItemRepositoryImpl">
<bean id="hibernateTemplate"
class="org.springframework.orm.hibernate3.HibernateTemplate">
org.hibernate.dialect.Oracle9Dialect
<bean id="dataSource"
class="org.apache.commons.dbcp.BasicDataSource"
destroy-method="close">
Sping配置文件
最開始的時候,服務代碼是這樣寫的
public BookItem findById(long bookItemId) {
BookItem bookItem = (BookItem) getHibernateTemplate().load(
BookItem.class, new Long(bookItemId));
return bookItem;
}
JUNIT測試代碼如下
public class BookItemRepositoryTest extends TestCase {
BookItemRepository bookItemRepo=null;
protected void setUp() throws Exception {
ApplicationContext context=new
ClassPathXmlApplicationContext("spring-context.xml");
bookItemRepo=(BookItemRepository)context.getBean("BookItemRepositoryImpl");
}
public void testFindById(){
BookItem item=bookItemRepo.findById(30);
assertEquals("123456", item.getBook().getISBN());
assertEquals("QiuHongBookStore", item.getBookStore().getName());
}
}
測試運行以后報下列錯誤
org.hibernate.LazyInitializationException: could not initialize proxy - no Session
at org.hibernate.proxy.AbstractLazyInitializer.initialize(AbstractLazyInitializer.java:57)
at org.hibernate.proxy.AbstractLazyInitializer.getImplementation(AbstractLazyInitializer.java:111)
at org.hibernate.proxy.pojo.cglib.CGLIBLazyInitializer.invoke(CGLIBLazyInitializer.java:150)
at domain.BookItem$$EnhancerByCGLIB$$2f924ddd.getBook(
at test.domain.repository.hibernate.BookItemRepositoryTest.testFindById(BookItemRepositoryTest.java:23)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at junit.framework.TestCase.runTest(TestCase.java:154)
at junit.framework.TestCase.runBare(TestCase.java:127)
at junit.framework.TestResult$1.protect(TestResult.java:106)
at junit.framework.TestResult.runProtected(TestResult.java:124)
at junit.framework.TestResult.run(TestResult.java:109)
at junit.framework.TestCase.run(TestCase.java:118)
at junit.framework.TestSuite.runTest(TestSuite.java:208)
at junit.framework.TestSuite.run(TestSuite.java:203)
at org.eclipse.jdt.internal.junit.runner.junit3.JUnit3TestReference.run(JUnit3TestReference.java:130)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:460)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:673)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:386)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:196)
開始我在HBM文件中,把
<class name="BookItem" table="T_BookItem" lazy="true".....
然后就正常了。但是這樣的代價是取消延遲加載,在load的時候直接進行加載。如果這樣的話,在對象很大的時候性能會很差。于是在網(wǎng)上查資料。這個主要是HibernateTemplate在加載的時候,采用是lazy 加載方法,因為HibernateTemplat 在load完成后自動的關(guān)閉了Session,所以造成了LazyInitializationException異常。
我最開始將代碼改成如下,也能解決這個問題。
public BookItem findById(long bookItemId) {
BookItem bookItem = (BookItem)getSession().load(
BookItem.class, new Long(bookItemId));
return bookItem;
}
但是這樣改我心里面沒有地,我擔心會不會存在Session沒有關(guān)閉或者泄漏的問題。
后來我聯(lián)想如果在Spring中增加事務控制,那么HibernateTemplate是不是就不會在調(diào)用完成后馬上Close Session了呢?(如果Close了Session,那事務是不是都需要Commit或RollBack了呢?,那這樣的話還談什么事務控制呢)。于是我進行了如下事務配置
<bean id="BookItemRepositoryImpl"
class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<bean
class="domain.repository.hibernate.BookItemRepositoryImpl">
然后代碼也進行了修改,執(zhí)行手工加載操作。
public BookItem findById(long bookItemId) {
BookItem bookItem = (BookItem) getHibernateTemplate().load(
BookItem.class, new Long(bookItemId));
getHibernateTemplate().initialize(bookItem); // 手工加載每個對象
getHibernateTemplate().initialize(bookItem.getBook()); // 手工加載每個對象
getHibernateTemplate().initialize(bookItem.getBookStore()); //手工加載每個對象
return bookItem;
}
再進行測試就OK了!
感謝各位的閱讀,以上就是“Hibernate Lazy加載問題怎么解決”的內(nèi)容了,經(jīng)過本文的學習后,相信大家對Hibernate Lazy加載問題怎么解決這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。