Hibernate Lazy 載入問題的解決

qiuhj1978發表於2010-12-17

今天在寫Spring+Hibernate的時候,遇到了Hibernate Lazy 載入的問題,後來經過調整配置解決了一個惱人的問題。現將解決方案和大家分享一下。




SEQBOOKID














Hbm檔案

class="domain.repository.hibernate.BookItemRepositoryImpl">





class="org.springframework.orm.hibernate3.HibernateTemplate">











classpath:domain/BookItem.hbm.xml
classpath:domain/BookStore.hbm.xml



true
true
false
false

org.hibernate.dialect.Oracle9Dialect




class="org.apache.commons.dbcp.BasicDataSource"
destroy-method="close">

oracle.jdbc.driver.OracleDriver


jdbc:oracle:thin:@192.168.10.28:1521:VIPDBZJ


scott


tiger

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檔案中,把

lazy="true".....

然後就正常了。但是這樣的代價是取消延遲載入,在load的時候直接進行載入。如果這樣的話,在物件很大的時候效能會很差。於是在網上查資料。這個主要是HibernateTemplate在載入的時候,採用是lazy 載入方法,因為HibernateTemplat 在load完成後自動的關閉了Session,所以造成了LazyInitializationException異常。

我最開始將程式碼改成如下,也能解決這個問題。

public BookItem findById(long bookItemId) {
BookItem bookItem = (BookItem)getSession().load(
BookItem.class, new Long(bookItemId));
return bookItem;
}

但是這樣改我心裡面沒有地,我擔心會不會存在Session沒有關閉或者洩漏的問題。

後來我聯想如果在Spring中增加事務控制,那麼HibernateTemplate是不是就不會在呼叫完成後馬上Close Session了呢?(如果Close了Session,那事務是不是都需要Commit或RollBack了呢?,那這樣的話還談什麼事務控制呢)。於是我進行了如下事務配置

class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">




class="domain.repository.hibernate.BookItemRepositoryImpl">







PROPAGATION_REQUIRED


然後程式碼也進行了修改,執行手工載入操作。

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了!

[@more@]

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/90000/viewspace-1043221/,如需轉載,請註明出處,否則將追究法律責任。

相關文章