文档章节

Struts2执行流程

郏高阳
 郏高阳
发布于 2011/12/29 21:43
字数 2580
阅读 1.5K
收藏 4

3 月,跳不动了?>>>

Struts2执行流程
1. web.xml 部署描述符
2. FilterDispatcher 实现StrutsStatics, Filter接口
(1)Filter:一个filter是一个对象用于执行过滤任务为每个请求资源(一个servlet或静态内容),或响应一个资源,或两者.过滤器执行过滤是在doFilter方法中.每个过滤器访问一个FilterConfig对象从中获取初始化参数,一个引用到ServletContext可以被使用,例如,在过滤器任务需要时装载资源.过滤器的配置在Web应用程序的部署描述符中.
init()-初始化过滤器,它的输入参数javax.servlet.FilterConfig的一个实例,可以这里初始化过滤要使用到的FilterConfig。这个方法由Web容器自动调用。
doFilter()-进行具体的过滤操作,这个方法以javax.servlet.ServletRequest请求信息, javax.servlet.ServletResponse响应信息,javax.servlet.FilterChain过滤链。过滤链,在Web应 用程序中所有的过滤器会构成一个链状,符合过滤条件的程序将会根据定义的顺序执行所有链中的过滤器。在这个方法中调用FilterChain的 doFilter(javax.servlet.ServletRequest, javax.servlet.SerletResponse)方法就可以传递到链中的下一个过滤器。
destory()-销毁过滤器,可以在这里释放使用完的资源,例如设置过滤器中FilterConfig为null。
(2)StrutsStatics: 定义Struts中使用的常量,常量用来获取和设置Action上下文以外的对象或其它集合.可以使用如下方式获取这些对象
ActionContext context = ActionContext.getContext();
HttpServletRequest request = (HttpServletRequest)context.get(HTTP_REQUEST);

(3) FilterConfig接口:一个过滤器配置对象,用于Servlet容器传递信息到过滤器在初始化期间.
BeanSelectionProvider类:选择框架实现的关键扩展点,用于装载属性常量.选择的实现是从container builder使用的名字定义关联属性,默认名为”struts”
例如:获取request
ActionContext.getContext().put(HTTP_REQUEST, request)
执行流程:
※ init(FilterConfig filterConfig)初始化过滤器
->createDispatcher(filterConfig); 创建调度器,返回带有过滤器参数Dispatcher
    ->执行Dispatcher的init方法,首先创建configurationManager,如果为null使用BeanSelectionProvider. DEFAULT_BEAN_NAME参数创建.
    -> 配置的初始化
//加载org/apache/struts2/default.properties文件
init_DefaultProperties(); // [1]
//加载格式相同的struts-default.xml,struts-plugin.xml,struts.xml
init_TraditionalXmlConfigurations(); // [2]
//可读取public static final String STRUTS_LOCALE = "struts.locale";配置
         init_LegacyStrutsProperties(); // [3]
   //加载Web.xml中的actionPackages配置
         init_ZeroConfiguration(); // [4]
   //读取configProviders参数,配置用户定义的provider
         init_CustomConfigurationProviders(); // [5]
   //Struts2.0.9中为空
         init_MethodConfigurationProvider();
   //初始化Filter中的config参数中指定的配置文件, initParams参数的获取是在Dispatcher创建时
         init_FilterInitParameters() ; // [6]
   //在BeanSelectionProvider中注册别名
         init_AliasStandardObjects() ; // [7]
   //读取public static final String STRUTS_I18N_RELOAD = "struts.i18n.reload";参数,
Container container = init_PreloadConfiguration();
//读取public static final String STRUTS_CONFIGURATION_XML_RELOAD = "struts.configuration.xml.reload";
         init_CheckConfigurationReloading(container);
         //初始化weblogic相关配置
         init_CheckWebLogicWorkaround(container);
-> String param = filterConfig.getInitParameter("packages");
         String packages = "org.apache.struts2.static template org.apache.struts2.interceptor.debugging";设置静态资源Prefixes
※->doFilter
   处理一个Action和它请求的静态资源,过滤器尝试配置请求到一个Action mapping.如果mapping找到了,Action的处理在dispatcher的serviceAction方法.如果Action处理失败,doFilter将会尝试去创建一个错误页面通过dispatcher.另外请求一个静态资源,资源将直接Copy到响应,使用适当的caching头设置.如果请求没有匹配一个Action mapping或一个静态资源页,那么将通过.
(1) getServletContext
    Servlet2.3规范中Servlet Context可以从Session中从新获取。不幸的是,一些版本的Weblogic不能够从新获取从Filter的配置中.因此,这个方法允许子类重新获取Servlet Context从其它源.
(2) prepareDispatcherAndWrapRequest(request, response)
   包装和返回给定的request,如果需要,作为处理multipart data包装给定的request.
1.dispatcher.prepare(request, response);
准备request,包括设置encoding和locale,使用注入的public static final String STRUTS_I18N_ENCODING = "struts.i18n.encoding"、public static final String STRUTS_LOCALE = "struts.locale"和public static final String STRUTS_DISPATCHER_PARAMETERSWORKAROUND = "struts.dispatcher.parametersWorkaround";(是否使用一个Servlet request参数,对一些版本的Weblogic是必需的)
2.request = dispatcher.wrapRequest(request, getServletContext());
   首先包装request,通过预先编码可以使用multipart/form-data参数访问的情况.
3. mapping = actionMapper.getMapping(request, dispatcher.getConfigurationManager());找到对应的Action映射.
4. 如果找不到就去查找静态资源.
可以设置public static final String STRUTS_SERVE_STATIC_CONTENT = "struts.serve.static";
5.然后是正常的chain.doFilter(request, response);请求传递
6. dispatcher.serviceAction(request, response, servletContext, mapping);
    装载映射的Acation类和调用适当的Action方法,或直接转向Result.这个方法首先创建Action上下文从给定的参数,并且装载一个ActionProxy从给定的Action名和namespace.然后,Action方法被直行和输出通道通过response对象.Action没有找到,发送到用户是通过Dispatcher的sendError方法,使用404返回码.所有的其它错误是通过抛出ServletException异常.
①ValueStack stack = (ValueStack) request.getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
如果已经存在这样的值栈,创建一个新的,拷贝和传递它为新的Action所使用.
② ActionProxy proxy = config.getContainer().getInstance(ActionProxyFactory.class).createActionProxy(
                     namespace, name, extraContext, true, false);
ActionProxy将会查阅框架的配置文件管理(初始化是从Struts.xml文件),下一步,ActionProxy创建一个ActionInvocation,它来负责命令模式的实现. 包括调用拦截器在调用Action自身之前.
③ proxy.setMethod(method);
   设置Action调用中要执行的方法.如果没有方法被指定,将会由Action的配置来提供.
设置Action中执行的方法.
④如果ActionMapping直接到一个result
if (mapping.getResult() != null) {
                 Result result = mapping.getResult();
                 result.execute(proxy.getInvocation());
             } else {
//没有配置result的情况
                 proxy.execute();
             }
⑤调用ActionProxy接口中的String execute() throws Exception;方法,它的实现类为DefaultActionProxy,在其中会调用retCode = invocation.invoke()方法.
⑥ActionInvocation接口的实现类DefaultActionInvocation的invoke方法首先调用拦截器栈,如果拦截器执行完,调用Action使用resultCode = invokeActionOnly();
⑦接下来迭代preResultListeners中的PreResultListener并执行其中的listener.beforeResult(this, resultCode);方法,在Action与Result之间.
⑧执行Result
if (proxy.getExecuteResult()) {
         executeResult();
        }
⑨到此serviceAction方法执行完毕,返回到FilterDispatcher的doFilter方法中
⑩释放本地线程和创建的DispatchListeners
public void destroy() {
         if (dispatcher == null) {
             LOG.warn("something is seriously wrong, Dispatcher is not initialized (null) ");
         } else {
             dispatcher.cleanup();
         }
     }
1. FilterDispatcher
(一)Struts的主要filter有四种不同的职责:
(1) 执行Action
这个Filter执行Action是通过查找ActionMapper来确定请求的URL将会调用的Action.如果mapper中指定,其它的Filter chain将会停止并且Action会被调用.这一点非常重要,它意味着像SiteMesh过滤器必需被放置在它之前或者将不会被修饰输出的Action.
(2) Cleaning up ActionContext (清理上下文)
这个Filter将自动为你清理ActionContext,确保不会出现内存泄漏.然而,有时会引起一些集成上的问题像使用SiteMesh.见ActionContextCleanUp来获取更多的信息处时这个问题.
(3) Serving static content(静态内容服务)
这个过滤器也服务于普通的静态内容,需要使用Struts中的很多部分,像JavaSctipt文件,CSS文件,等等.它的工作通过查找请求中的/struts/*,并且映射值在”/struts/”后面的常用包,它是可选的在你的classpath中.默认情况下,下面的包将会自动被搜索:
★org.apache.struts2.static
★template
这意味着你可以请求/struts/xhtml/styles.css和XHTML UI主题的默认stylesheet将会被返回.同样的很多AJAX UI组件需要的JavaScript文件也可以在org.apache.struts2.static包中找到.如果你希望添加额外的包被查找到,你可以增加一个逗号分隔的列表在过滤器的被始化参数”packages”中.注意,这样将暴露了你的包中的敏感信息,像访问数据库的属性文件.
(4) Kicking off XWork's interceptor chain for the request lifecycle
(二)这个Filter必需映射到所有的请求.除非你道它在做什么,通常映射到/*种URL模式
(三)这个Filter支持如下init-params:
    ★config---------------一个逗号分隔的XML配置文件将会被装载
    ★actionPackages----一个逗号分隔的Java Action包将会被扫描.
    ★configProviders---一逗号分隔的实现com.opensymphony.xwork2.config.ConfigurationProvider 接口的Java类列表,用于构建com.opensymphony.xwork2.config.Configuration。
    ★*-------------任何其它的参数,作为框架的常量来对待.
(四)使用定制的Dispatcher, createDispatcher方法可以被覆盖通过子类.

通配符映射
随着应用程序的增大,Action的数量也会增多.通配符可以用来组合简单的映射到多个通用映射.
最好的解释通配符的方法是通过一个例子和它是怎样工作的来展示.这个例子修改一个习惯上用通配符来匹配所有以/edit开头的页面;
<!-- Generic edit* mapping -->
<action
     name="/edit*"
     class="org.apache.struts.webapp.example.Edit{1}Action">
     <result
         name="failure"
         path="/mainMenu.jsp"/>
     <result
         path="/\{1\}.jsp"/>
</action>
在路径中的“*”属性允许映射匹配请求的URI /editSubscription,editRegistration,或其它URI以/edit开头的,然而/editSubscription/add将不会被匹配.URI的部分匹配通过通配符,被替代的是Action映射中的变量属性和Action结果替换{1}.为了其它的请求,框架将查看Action映射和Action result中包括的新值.
映射的匹配依赖于请求,依次出现在框架的配置文件中。如果一个或多个模式匹配最后一个模式将得到,所以较少的指定模式,必需呈现多个指向同一个之前.然而,如果请求的URL可以被匹配依赖于一个路径不需要任何通配置,没有通配置被执行,同时顺序也不重要了.
通配符模式可以包含一个或多个下面的指定标记:
*匹配一个或多个字符包括斜线(‘/’)字符.
** 匹配一个或多个字符除了斜线(‘/’)字符.
\字符,反斜杠字符用来作为一个脱字符连接.因此*匹配字符星(‘*’),和匹配字符反斜框(‘\’).
在Action Mapping和Action Result,通配符匹配值可以被该问使用标记{N},N从1到9指示通配符匹配值到被替代属性 .全部的请求URI可以被该问用{0}标记.
同样的,Action Mapping属性(设置使用<set-property key=”foo” value=”bar”>语法)将接受通配符匹配字符串在他们的值属性.
像Action Mapping,Action result属性(设置使用<set-property key=”foo” value=”bar”>语法)将接受通配符匹配字符串在他们的值属性.
例子代码:
<action name="Login_*" method="{1}" class="example.Login">
             <result name="input">/example/Login.jsp</result>
             <result type="redirect-action">Menu</result>
         </action>

         <action name="*" class="example.ExampleSupport">
             <result>/example/{1}.jsp</result>
         </action>

© 著作权归作者所有

郏高阳

郏高阳

粉丝 103
博文 153
码字总数 112334
作品 1
黄浦
程序员
私信 提问
加载中

评论(1)

52java
52java
博主排一下版吧,文章还是很不错滴,先收藏了,以后慢慢看
【Java框架】Java EE框架常见的面试题

一、什么是Spring? 1、Spring的核心是一个轻量级(Lightweight)的容器(Container)。 2、Spring是实现IoC(Inversion of Control)容器和非入侵性(No intrusive)的框架。 3、Spring提供...

林元煌
2017/07/31
0
0
工作流软件产品集成struts2框架

工作流软件产品只负责业务流程的流转,流程节点上业务模块的办理,以及监控业务流程的执行。通常一套工作流管理系统,会包含流程引擎,流程设计器,流程管理中心,表单设计器,自定义表单系统...

长平狐
2012/10/11
141
0
SSH-Struts第三弹:传智播客视频教程第一天上午的笔记

一、 框架概述 1、三大框架 : 是企业主流 JavaEE 开发的一套架构 Struts2 + Spring + Hibernate 2、 什么是框架?为什么要学框架 ? 框架 是 实现部分功能的代码 (半成品),使用框架简化企...

candy-yun
2014/03/26
0
0
Java常见面试题汇总-----------Java框架

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 https://blog.csdn.net/zengxiantao1994/article/details/89475719 47、struts2的执行流程?...

知行流浪
2019/04/23
0
0
struts1-2,springMVC原理基本对比(单例,多例)-servlet与filter区别

最近做项目用到了struts2,之前一直是用struts1和springMVC。感觉到了struts2从很大程度上和这两个还是有很大区别的,所以今天搜集了些资料,给他们做一下对比。 Struts1官方已经停止更新,现...

tantexian
2016/06/21
447
0

没有更多内容

加载失败,请刷新页面

加载更多

为什么只能在头文件中实现模板? - Why can templates only be implemented in the header file?

问题: Quote from The C++ standard library: a tutorial and handbook : 引用来自C ++标准库:教程和手册 : The only portable way of using templates at the moment is to implement t......

javail
今天
19
0
Gradle 6 针对已有的构建如何创建一个构建扫描

有关构建扫描的定义为: 构建扫描(build scan)是一个中心化并且可以共享的构建记录。这个构建记录通常能够告诉在构建中发生了什么并且为什么会发生。 通过应用构建扫描插件到你的项目中,你...

honeymoose
今天
17
0
C语言动态内存分配:(一)malloc/free的实现及malloc实际分配/释放的内存

一、malloc/free概述 malloc是在C语言中用于在程序运行时在堆中进行动态内存分配的库函数。free是进行内存释放的库函数。 1、函数原型 #include <stdlib.h> void *malloc( size_t size ); v...

shzwork
今天
17
0
什么是JavaBean? - What is a JavaBean exactly?

问题: I understood, I think, that a "Bean" is a Java class with properties and getters/setters. 我认为,“ Bean”是具有属性和getter / setter的Java类。 As much as I understand,......

技术盛宴
今天
27
0
深圳援鄂最后一批工作人员归来,88万元关爱金发放至85人

中国公益在线3月31日深圳讯 深圳援鄂最后一批工作人员归来......深圳市民政局、深圳市卫健委和深圳市慈善会发起了“深爱战疫天使基金”项目,联合龙华区慈善会和 永贤慈善基金会,进行第二次...

传承天下融媒体中心
今天
15
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部