`

Hibernate/Spring/Struts架构使用OpenSessionInView的

阅读更多

今天有一个朋友问了我一个问题,他使用的是Hibernate/Spring/Struts架构,配置使用SpringOpenSessionInView Filter,但是发现不生效,lazy的集合属性在页面访问的时候仍然报session已经关闭的错误。我和他一起检查了所有的配置和相关的代码,但是没有发现任何问题。经过调试发现,应用程序使用的SessionOpenSessionInView Filter打开的Session不是同一个,所以OpenSessionInView模式没有生效,但是为什么他们不使用同一个Session呢?

检查了一遍Spring的相关源代码,发现了问题的根源:

通常在Web应用中初始化Spring的配置,我们会在web.xml里面配置一个Listener,即:

xml代码


        <listener>
                <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
        </listener>

<o:p> </o:p>



如果使用Struts,那么需要在Struts的配置文件struts-config.xml里面配置一个SpringpluginContextLoaderPlugIn

实际上ContextLoaderListenerContextLoaderPlugIn的功能是重叠的,他们都是进行Spring配置的初始化工作的。因此,如果你不打算使用OpenSessionInView,那么你并不需要在web.xml里面配置ContextLoaderListener

好了,但是你现在既需要Struts集成Spring,又需要OpenSessionInView模式,问题就来了!

由于ContextLoaderListenerContextLoaderPlugIn功能重叠,都是初始化Spring,你不应该进行两次初始化,所以你不应该同时使用这两者,只能选择一个,因为你现在需要集成Struts,所以你只能使用ContextLoaderPlugIn

但是令人困惑的是,ContextLoaderListenerContextLoaderPlugIn有一个非常矛盾的地方!

ContextLoaderListener
初始化spring配置,然后把它放在ServletContext对象里面保存:

java代码


servletContext.setAttribute(
                                        WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);

<o:p> </o:p>



请注意,保存的对象的keyWebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE

但是ContextLoaderPlugIn初始化spring配置,然后把它放在ServletContext对象里面保存:

java代码



String attrName = getServletContextAttributeName();
getServletContext().setAttribute(attrName, wac);

<o:p> </o:p>



这个attrNameWebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE名字是不一样的!

如果仅仅是名字不一样,问题还不大,你仍然可以放心使用ContextLoaderPlugIn,但是当你使用OpenSessionInView的时候,OpenSessionInViewFilter是使用哪个key取得spring配置的呢?

java代码


WebApplicationContext wac =
                                WebApplicationContextUtils.getRequiredWebApplicationContext(getServletContext());

<o:p> </o:p>



 

*********************************** 

***********************************

 

我原来用struts/spring/hibernate的时候同样使用OpenSessionInView,但是似乎没有robbin所说的问题啊。而且我在使用的时候,是ContextLoaderListenerContextLoaderPlugIn一起用的。整个配置如下:
1.
首先是web.xml

java代码



        <filter>
        <filter-name>OpenSessionInViewFilter</filter-name>
        <filter-class>org.springframework.orm.hibernate.support.OpenSessionInViewFilter</filter-class>
    </filter>
   
    <filter-mapping>
        <filter-name>OpenSessionInViewFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
   
    <listener>
                <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
        </listener>

......

<o:p> </o:p>



2.
然后是struts-config.xml

java代码



<plug-in className="org.springframework.web.struts.ContextLoaderPlugIn">
        <set-property property="contextConfigLocation"
                                  value="/WEB-INF/action-servlet.xml"
        />
</plug-in>

<o:p> </o:p>



其余部分省略。

在上述配置下,使用OpenSessionInView似乎没有问题。

不知道robbin所说的ContextLoaderListenerContextLoaderPlugIn不应该同时使用是不是做得是如下的配置:(struts-config.xml

java代码



<plug-in
className="org.springframework.web.struts.ContextLoaderPlugIn">
<set-property property="contextConfigLocation"
value="/WEB-INF/applicationContext.xml,
/WEB-INF/action-servlet.xml"/>
</plug-in>

<o:p> </o:p>



我尝试了一下,用这种配置时,OpenSessionInView的确失效了。

我猜想,原因大概是这样:struts的这个plugIn,可能只是为了整合一个action-servlet.xml,将action-servlet.xml中的定义当作Springbean来使用,因此,在保存时,只要有action-servlet.xml的配置,就被保存到robbin所提到的那个attrName中,而不是WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE中,所以,OpenSessionInView是取不到这个配置的。

那么这个配置什么时候被取到呢?直觉告诉我,可能是和ActionProxy有关。于是,查看了org.springframework.web.struts.DelegatingActionProxy的源码,果然:

java代码



/**
        * Return the delegate Action for the given mapping.
        * <p>The default implementation determines a bean name from the
        * given ActionMapping and looks up the corresponding bean in the
        * WebApplicationContext.
        * @param mapping the Struts ActionMapping
        * @return the delegate Action
        * @throws BeansException if thrown by WebApplicationContext methods
        * @see #determineActionBeanName
        */
        protected Action getDelegateAction(ActionMapping mapping) throws BeansException {
                WebApplicationContext wac = getWebApplicationContext(getServlet(), mapping.getModuleConfig());
                String beanName = determineActionBeanName(mapping);
                return (Action) wac.getBean(beanName, Action.class);
        }

        /**
        * Fetch ContextLoaderPlugIn's WebApplicationContext from the
        * ServletContext, containing the Struts Action beans to delegate to.
        * @param actionServlet the associated ActionServlet
        * @param moduleConfig the associated ModuleConfig
        * @return the WebApplicationContext
        * @throws IllegalStateException if no WebApplicationContext could be found
        * @see DelegatingActionUtils#getRequiredWebApplicationContext
        * @see ContextLoaderPlugIn#SERVLET_CONTEXT_PREFIX
        */
        protected WebApplicationContext getWebApplicationContext(
                        ActionServlet actionServlet, ModuleConfig moduleConfig) throws IllegalStateException {
                return DelegatingActionUtils.getRequiredWebApplicationContext(actionServlet, moduleConfig);
        }

<o:p> </o:p>



 

来源:http://forum.iteye.com/viewtopic.php?t=15057

仔细看其中的取wac的代码,它并不是从WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE取的wac

由此,我相信,除了robbin讲的修改源码以外,同时使用ContextLoaderListenerContextLoaderPlugIn,但是不要在ContextLoaderPlugIn里面加入applicationContext.xml,只要加入你的action-servlet.xml,我相信,同样也可以非常流畅的使用OpenSessionInView
显然,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);

或者修改OpenSessionInViewFilter,让它按照attrName去取得spring配置。
分享到:
评论
3 楼 favor 2008-11-26  
明白了,谢过!
2 楼 guojiangbo 2008-10-13  
如果spring的配置文件除action-servlet.xml外有多个,其它的怎末配置?在哪里配置?
1 楼 bluemusic 2008-08-20  
我想问问这样配置后能正常运行么

相关推荐

    SSH整合 struts+hibernate+spring

    ### SSH整合 struts+hibernate+spring #### 一、SSH整合概述 在软件开发领域,特别是Java Web开发中,Struts、Spring 和 Hibernate 被广泛地应用于构建复杂的应用程序。这三大框架各有所长,分别在不同的层面发挥...

    Struts Spring Hibernate 整合 OpenSessionInView 例子

    2.通过spring注入dao到 action 3.通过 open session in view filter 支持 延迟加载 4.在页面上通过 jstl 很优雅的获取数据 5.通过 spring aop(aspectJ) 声明事务 6.通过formular 映射参数表,指定两个死的变量

    spring管理struts和hibernate

    ### Spring管理Struts与Hibernate详解 #### 一、Spring整合Struts 在Web开发中,Struts是一个基于MVC设计模式的开源框架,它能够帮助开发者构建可维护性高、结构清晰的应用系统。Spring框架则提供了强大的依赖注入...

    struts2+hibernate+spring

    根据提供的文件信息,我们可以推断出这是一篇关于整合Struts2、Hibernate和Spring框架的文章。下面将基于这些信息详细阐述这些技术的关键知识点。 ### Struts2+Hibernate+Spring框架整合 #### 一、概述 Struts2、...

    Struts+Spring+Hibernate开发实例.pdf

    标题和描述中指出的文档《Struts+Spring+Hibernate开发实例.pdf》包含了关于这三个流行的Java开源框架结合使用的示例代码和相关知识点。Struts负责视图与控制器部分,Spring负责业务逻辑层及依赖注入,Hibernate负责...

    SSH整合示例项目 hibernate5.2.3+struts2.5.2+spring4.3.3.zip

    SSH整合是Java Web开发中的一种经典技术栈,包括Spring、Struts和Hibernate三个核心框架。这个项目是一个基于最新版本SSH的示例应用,具体为Hibernate 5.2.3、Struts 2.5.2和Spring 4.3.3的集成。以下是关于SSH整合...

    struts+hibernate,spring面试题

    Struts、Hibernate和Spring(SSH)是Java Web开发中经典的三大框架,它们分别负责MVC模式中的模型(Model)、视图(View)和控制器(Controller)部分。在面试中,了解这些框架的核心概念和实际应用是非常重要的。 ...

    Struts+spring+hibernate面试资料

    例如,使用Struts2作为前端框架处理用户的请求和显示,使用Spring进行依赖注入和服务层的管理,而Hibernate则负责数据的持久化。这样的组合可以充分发挥各框架的优势,构建出高效、稳定的应用系统。 综上所述,...

    struts-2.3.24.1+spring4.2.1+hibernate5.0

    Struts 2.3.24.1、Spring 4.2.1 和 Hibernate 5.0 是三个在 Java 开发领域中广泛使用的开源框架,它们分别负责 Web 应用的 MVC(Model-View-Controller)架构、依赖注入与服务管理以及对象关系映射(ORM)。...

    struts+spring+hibernate面试题

    在Java Web开发中,Struts、Spring和Hibernate是最常见的三大框架,它们...以上就是关于Struts、Spring和Hibernate面试题的详细解析,涵盖了这三大框架的关键概念和使用技巧,对于理解和解答相关面试问题非常有帮助。

    struts spring hibernate面试题

    ### Struts、Spring、Hibernate 面试题详解 #### 1. Action 是否线程安全?如果不是,如何保证 Action 的线程安全? - **答案**:通常情况下,Struts 的 Action 不是线程安全的。因为同一个 Action 实例可能会被多...

    struts+spring+hibernate面试题.doc

    标题和描述指向的是一个关于Struts、Spring和Hibernate技术栈的面试题集合,这份文档旨在帮助准备面试的软件工程师理解并复习这些关键技术点。以下是基于给定内容的关键知识点的详细阐述: ### 1. Action的线程安全...

    收集的struts+spring+hibernate面试题.doc

    ### Struts + Spring + Hibernate 面试题解析 #### 1. AOP 和 IOC 的概念以及在 Spring 中的应用 **AOP(Aspect Oriented Programming,面向切面编程)**: 是一种编程思想,它允许程序员定义“切面”来封装那些...

    struts2+hirbate+spring面试题

    Struts2、Spring和Hibernate(SSH)是Java企业级开发中的三大主流框架,它们共同构建了一个强大的MVC(模型-视图-控制器)架构。在面试中,掌握这些框架的基本概念、工作原理以及最佳实践至关重要,尤其是对于刚踏入...

    收集的struts+spring+hibernate面试题借鉴.pdf

    Struts、Spring和Hibernate是Java Web开发中...以上是对Struts、Spring和Hibernate相关面试题的详细解答,涵盖了它们的主要概念、功能和使用方法。在实际项目中,这三大框架的集成使用能极大地提高开发效率和代码质量。

    SSH全注解环境搭建

    SSH框架是指Struts2、Spring、Hibernate这三种技术的组合,常被用于Java Web开发中。全注解配置相较于XML配置更加简洁,易于维护,因此在现代Web项目开发中越来越受到青睐。本文将详细介绍如何搭建一个基于全注解...

    Spring提供的CharacterEncoding和OpenSessionInView功能

    SSH框架结合了Struts的MVC设计模式、Spring的依赖注入和事务管理以及Hibernate的持久化能力,为Java Web开发提供了强大的支持。然而,随着Spring Boot的兴起,SSH框架的使用逐渐减少,更多地转向了Spring Boot的开箱...

    SSH项目整合示例【OpenSessionInView】所用到的jar包

    SSH是Java Web开发中的一个流行框架组合,由Struts、Hibernate和Spring三个组件构成。这个框架集成提供了模型-视图-控制器(MVC)架构,数据持久化,以及依赖注入和事务管理等功能,大大简化了企业级应用的开发工作...

    Spring的学习笔记

    - **Struts2,Hibernate,Spring**:确保引入了所有必要的jar包,包括Struts2的action、result、filter等,Hibernate的session工厂和日志库,以及Spring的相关模块。 这些知识点构成了Spring框架的基础,理解和掌握...

Global site tag (gtag.js) - Google Analytics