我使用的是Hibernate/Spring/Struts架構,配置使用Spring的OpenSessionInViewFilter,但是發現這個filter根本就不生效,lazy的集合屬性在頁面訪問的時候仍然報session已經關閉的錯誤。檢查了所有的配置和相關的代碼,但是沒有發現任何問題。經過調試發現,應用程式使用的Session和OpenSessionInView Filter開啟的Session不是同一個,所以OpenSessionInView模式沒有生效,但是為什麼他們不使用同一個Session呢?
檢查了一遍Spring的相關原始碼,發現了問題的根源:
通常在Web應用中初始化Spring的配置,我們會在web.xml裡面配置一個Listener,即:
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
如果使用Struts,那麼需要在Struts的設定檔struts.xml裡面配置一個Spring的plugin:ContextLoaderPlugIn。
<plug-in className="org.springframework.web.struts.ContextLoaderPlugIn">
<set-property property="contextConfigLocation" value="/WEB-INF/applicationContext.xml" />
</plug-in>
實際上ContextLoaderListener和ContextLoaderPlugIn的功能是重疊的,他們都是進行Spring配置的初始化工作的。因此,如果你不打算使用OpenSessionInView,那麼你並不需要在web.xml裡面配置ContextLoaderListener。
好了,但是你現在既需要Struts整合Spring,又需要OpenSessionInView模式,問題就來了!
由於ContextLoaderListener和ContextLoaderPlugIn功能重疊,都是初始化Spring,你不應該進行兩次初始化,所以你不應該同時使用這兩者,只能選擇一個,因為你現在需要整合Struts,所以你只能使用ContextLoaderPlugIn。
但是令人困惑的是,ContextLoaderListener和ContextLoaderPlugIn有一個非常矛盾的地方!
ContextLoaderListener初始化spring配置,然後把它放在ServletContext對象裡面儲存:
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
請注意,儲存的對象的key是WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE!
但是ContextLoaderPlugIn初始化spring配置,然後把它放在ServletContext對象裡面儲存:
String attrName = getServletContextAttributeName();
getServletContext().setAttribute(attrName, wac);
這個attrName和WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE名字是不一樣的!
如果僅僅是名字不一樣,問題還不大,你仍然可以放心使用ContextLoaderPlugIn,但是當你使用OpenSessionInView的時候,OpenSessionInViewFilter是使用哪個key取得spring配置的呢?
WebApplicationContext wac =WebApplicationContextUtils.getRequiredWebApplicationContext(getServletContext(););
顯然,OpenSessionInViewFilter是按照WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE這個key去拿spring配置的!
ContextLoaderPlugIn儲存spring配置的名字叫做attrName;
ContextLoaderListener儲存spring配置的名字叫做:WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE;
而OpenSessionInView是按照WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE這個名字去取得spring配置的!
而你的應用程式卻是按照attrName去取得spring的配置的!
所以,OpenSessionInView模式失效!
解決辦法:
修改ContextLoaderPlugIn代碼,在getServletContext().setAttribute(attrName, wac);這個地方加上一行代碼:
getServletContext().setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, wac);
本文出自 “想要跨越,就要付出更多” 部落格,請務必保留此出處http://andywuchuanlong.blog.51cto.com/4898493/1305320