OpenSessionInViewFilter問題

web.xml原始配置:
 <!-- 過濾spring中對於hibernate的session關閉管理 -->
 <filter>
  <filter-name>hibernateFilter</filter-name>
  <filter-class>
   org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
  </filter-class>
 </filter>
自己寫的serviceImpl.java文件中的保存更新方法(我所出現問題的位置是:多行提交的方法),在運行時總報錯。如下:

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

後來在網上狂搜解決方案,將web.xml文件改爲如下:
 <!-- 過濾spring中對於hibernate的session關閉管理 -->
 <filter>
     <filter-name>hibernateFilter</filter-name>
        <filter-class>
            org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
        </filter-class>
     <init-param>       
        <param-name>singleSession</param-name> 
        <param-value>false</param-value>    
     </init-param>
 </filter>
上面的異常解決了,但又報出新的異常,如下:
org.hibernate.HibernateException: Illegal attempt to associate a collection
with two open sessions
解決這個問題的辦法就是要把singleSession的值改爲true
<init-param>       
        <param-name>singleSession</param-name> 
        <param-value>true</param-value>   
</init-param>

我無奈了,這兩個錯誤就好像是相互的,只能解決一個。。。
從網上搜。。還真我出現的這咱情況。。結果如下:
說明一下Open Session in View的作用,就是允許在每次的整個request的過程中使用同一個hibernate session,可以在這個request任何時期lazy loading數據。
如果是singleSession=false的話,就不會在每次的整個request的過程中使用同一個hibernate session,而是每個數據訪問都會產生各自的seesion,等於沒有Open Session in View。
OpenSessionInViewFilter默認是不會對session 進行flush的,並且flush mode 是 never
代碼:
    protected Session getSession(SessionFactory sessionFactory) throws DataAccessResourceFailureException {
       Session session = SessionFactoryUtils.getSession(sessionFactory, true);
       session.setFlushMode(FlushMode.NEVER);
       return session;
    }
看getSession的方式就知道,把flush mode 設爲FlushMode.NEVER,這樣就算是commit的時候也不會session flush,
如果想在完成request過程中更新數據的話, 那就需要先把flush model設爲FlushMode.AUTO,再在更新完數據後flush.

OpenSessionInView默認的FlushMode爲
代碼:

FlushMode.NEVER


可以採用在寫保存更新刪除代碼的時候手動更改FlushMode
代碼:

         this.getHibernateTemplate().execute(new HibernateCallback() {
             public Object doInHibernate(Session session) throws HibernateException {
                 session.setFlushMode(FlushMode.AUTO);
                 session.save(user);
                 session.flush();
                 return null;
             }
         });


但是這樣做太繁瑣了,第二種方式是採用spring的事務聲明
代碼:

     <bean id="baseTransaction" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"
           abstract="true">
         <property name="transactionManager" ref="transactionManager"/>
         <property name="proxyTargetClass" value="true"/>
         <property name="transactionAttributes">
             <props> 
                 <prop key="load*">PROPAGATION_REQUIRED,readOnly</prop>
                 <prop key="save*">PROPAGATION_REQUIRED</prop>
                 <prop key="add*">PROPAGATION_REQUIRED</prop>
                 <prop key="update*">PROPAGATION_REQUIRED</prop>
                 <prop key="remove*">PROPAGATION_REQUIRED</prop>
             </props>
         </property>
     </bean>
代碼:

     <bean id="userService" parent="baseTransaction">
         <property name="target">
             <bean class="com.phopesoft.security.service.impl.UserServiceImpl"/>
         </property>
     </bean>


太巧了,我們的框架就採用了這位前輩所說的第二種方案,但是爲什麼我把配置文件改成他說的樣式還是不行呢?
鬱悶中驚奇發現,不是我所有的多行提交都出問題,而只是個別的。經過一翻考慮後,確定自己寫的方法體沒有
問題了,那麼就是方法名了,才發現,還真是方法名的問題。
原來自己寫的serviceImpl.java文件的方法名要用“load”“save”“add”“update”“remove”這些詞開頭,這個就好像是通過這個bean-service.xml文件管理方法名一樣,超出這個範圍了,hibernate自身的作用就發揮不出來了。
由於自己對spring,hibernate的瞭解不深,暫時先這樣理解。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章