真善美

          勇于承擔責任; 如果你做不到,要提前跟別人說明,不要等到事情發生后再解釋。

            BlogJava :: 首頁 ::  :: 聯系 :: 聚合  :: 管理 ::
            55 隨筆 :: 0 文章 :: 29 評論 :: 0 Trackbacks

          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的了解不深,暫時先這樣理解。

          posted on 2007-08-28 17:17 真善美 閱讀(9010) 評論(5)  編輯  收藏 所屬分類: Spring學習
          主站蜘蛛池模板: 五华县| 伊宁市| 万荣县| 张家口市| 石屏县| 凯里市| 松潘县| 荔浦县| 枞阳县| 遂昌县| 房产| 林芝县| 宜昌市| 玉田县| 留坝县| 呼玛县| 二连浩特市| 宁安市| 密云县| 天门市| 星座| 柘荣县| 施秉县| 明水县| 汶上县| 钦州市| 陇川县| 左权县| 宿迁市| 乐陵市| 县级市| 平遥县| 乌拉特中旗| 信丰县| 竹溪县| 长岭县| 白河县| 南川市| 建宁县| 文安县| 江北区|