open session in view 研究

在没有使用Spring提供的Open Session In View情况下,因需要在service(or Dao)层里把session关闭,所以lazy loading 为true的话,要在应用层内把关系集合都初始化,如 company.getEmployees(),否则Hibernate抛session already closed Exception;    Open Session In View提供了一种简便的方法,较好地解决了lazy loading问题.     
    它有两种配置方式OpenSessionInViewInterceptor和OpenSessionInViewFilter(具体参看SpringSide),功能相同,只是一个在web.xml配置,另一个在application.xml配置而已。     
     Open Session In View在request把session绑定到当前thread期间一直保持hibernate session在open状态,使session在request的整个期间都可以使用,如在View层里PO也可以lazy loading数据,如 ${ company.employees }。当View 层逻辑完成后,才会通过Filter的doFilter方法或Interceptor的postHandle方法自动关闭session。 
     OpenSessionInViewInterceptor配置 

Xml代码 
  1.     
  2. <beans>   
  3. <bean name="openSessionInViewInterceptor" class="org.springframework.orm.hibernate3.support.OpenSessionInViewInterceptor">   
  4. <property name="sessionFactory">   
  5. <ref bean="sessionFactory"/>   
  6. </property>   
  7. </bean>   
  8. <bean id="urlMapping" class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">   
  9. <property name="interceptors">   
  10. <list>   
  11. <ref bean="openSessionInViewInterceptor"/>   
  12. </list>   
  13. </property>   
  14. <property name="mappings">   
  15. ...   
  16. </property>   
  17. </bean> ... </beans>   


OpenSessionInViewFilter配置

Xml代码 
  1.    
  2. <web-app>   
  3. ...   
  4. <filter>   
  5. <filter-name>hibernateFilter</filter-name>   
  6. <filter-class> org.springframework.orm.hibernate3.support.OpenSessionInViewFilter </filter-class>   
  7. <!-- singleSession默认为true,若设为false则等于没用OpenSessionInView -->   
  8. <init-param>   
  9. <param-name>singleSession</param-name>   
  10. <param-value>true</param-value>   
  11. </init-param>   
  12. </filter> ... <filter-mapping>   
  13. <filter-name>hibernateFilter</filter-name>   
  14. <url-pattern>*.do</url-pattern>   
  15. </filter-mapping> ... </web-app>   


    很多人在使用OpenSessionInView过程中提及一个错误: 
org.springframework.dao.InvalidDataAccessApiUsageException: Write operations are not allowed in read-only mode (FlushMode.NEVER) - turn your Session into FlushMode.AUTO or remove 'readOnly' marker from transaction definition 
    看看OpenSessionInViewFilter里的几个方法

Java代码 
  1.     
  2.   
  3. protected void doFilterInternal(HttpServletRequest request,   
  4.         HttpServletResponse response,  
  5.         FilterChain filterChain) throws ServletException, IOException {   
  6.         SessionFactory sessionFactory = lookupSessionFactory();   
  7.         logger.debug("Opening Hibernate Session in OpenSessionInViewFilter");   
  8.         Session session = getSession(sessionFactory);   
  9.         TransactionSynchronizationManager.bindResource(    
  10.                 sessionFactory, new SessionHolder(session));   
  11.         try {    
  12.             filterChain.doFilter(request, response);   
  13.             }   
  14.         finally {   
  15.             TransactionSynchronizationManager.unbindResource(sessionFactory);   
  16.         logger.debug("Closing Hibernate Session in OpenSessionInViewFilter");   
  17.         closeSession(session, sessionFactory);   
  18.         }  
  19. }   
  20. protected Session getSession(SessionFactory sessionFactory)  
  21.                    throws DataAccessResourceFailureException {   
  22.         Session session = SessionFactoryUtils.getSession(sessionFactory, true);   
  23.         session.setFlushMode(FlushMode.NEVER);   
  24.         return session;  
  25. }  
  26. protected void closeSession(Session session,   
  27.         SessionFactory sessionFactory)throws CleanupFailureDataAccessException {   
  28.     SessionFactoryUtils.closeSessionIfNecessary(session, sessionFactory);  
  29. }  


     关于绑定session的方式,通过看spring里TransactionSynchronizationManager的实现,发现:它维护一个 java.lang.ThreadLocal类型的resources,resources负责持有线程局部变量,这里resources持有的是一个 HashMap,通过TransactionSynchronizationManager.bindResource()方法在map里绑定和线程相关的所有变量到他们的标识上,包括如上所述的绑定在sessionFactory上的线程局部session。sessionHolder只不过是存放可以 hold一个session并可以和transtaction同步的容器。可以看到 OpenSessionInViewFilter在getSession的时候,会把获取回来的session的flush mode 设为FlushMode.NEVER。然后把该sessionFactory绑定到 TransactionSynchronizationManager,使request的整个过程都使用同一个session,在请求过后再接除该 sessionFactory的绑定,最后closeSessionIfNecessary根据该session是否已和transaction绑定来决定是否关闭session。绑定以后,就可以防止每次不会新开一个Session呢?看看HibernateDaoSupport的情况:

Java代码 
  1.    
  2. public final void setSessionFactory(SessionFactory sessionFactory) {   
  3. this.hibernateTemplate = new HibernateTemplate(sessionFactory);   
  4. }    
  5. rotected final HibernateTemplate getHibernateTemplate() {  
  6. return hibernateTemplate;    
  7.            


     我们的DAO将使用这个template进行操作.

Java代码 
  1.      
  2. public abstract class BaseHibernateObjectDao   
  3.                 extends HibernateDaoSupportimplements BaseObjectDao {       
  4. protected BaseEntityObject getByClassId(final long id) {                  
  5. BaseEntityObject obj =(BaseEntityObject)getHibernateTemplate().execute(new HibernateCallback() {                          
  6. public Object doInHibernate(Session session)   
  7.          throws HibernateException{                                      
  8.  return session.get(getPersistentClass(),new Long(id));                  
  9.        }                  
  10.     }  
  11. );                  
  12. return obj;        
  13. }       
  14. public void save(BaseEntityObject entity) {                    
  15.        getHibernateTemplate().saveOrUpdate(entity);       
  16. }       
  17. public void remove(BaseEntityObject entity) {                
  18. try {                       
  19.        getHibernateTemplate().delete(entity);                
  20. catch (Exception e) {                        
  21.        throw new FlexEnterpriseDataAccessException(e);               
  22.        }        
  23. }         
  24. public void refresh(final BaseEntityObject entity) {                 
  25.        getHibernateTemplate().execute(new HibernateCallback(){                            
  26.             public Object doInHibernate(Session session)   
  27.            throws HibernateException   {                                  
  28.                  session.refresh(entity);                                        
  29.                  return color: #7f
分享到:
评论

你可能感兴趣的:(DAO,spring,Hibernate,Web,bean)