11.struts2的基本原理(struts2笔记)

这里我们只是讲一下最基本的原理,也就是当页面请求提交过来之后struts2是怎样处理的。
在网上找了一张图,地址是:

http://blog.csdn.net/wuwenxiang91322/article/details/11070513

struts2原理图.jpg

其中控制器就是StrutsPrepareAndExecuteFilter或FilterDispatcher。在Struts2.1以前调用FilterDispatcher,Struts2.1以后调用StrutsPrepareAndExecuteFilter,控制器通过ActionMapper得到action的相关信息,通过ActionProxy去读取相关配置文件,主要过程是在ActionInvocation(即DefaultActionInvocation)和实际action之间。我们就从控制器开始。

首先我们在org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter这个类的doFilter方法开头设置一个断点,使用debug进行跟踪。在此方法中我们可以看到这样一段:

 ActionMapping mapping = prepare.findActionMapping(request, response, true);
               if (mapping == null) {
                    boolean handled = execute.executeStaticResourceRequest(request, response);
                    if (!handled) {
                        chain.doFilter(request, response);
                    }
                } else {
                    execute.executeAction(request, response, mapping);
                }

说明:这里表示如果mapping为空表示没有找到相应的action,否则表示找到了相应的action,找到之后执行

execute.executeAction(request, response, mapping);

方法,我们跟进去,进去之后会发现执行的是方法:

dispatcher.serviceAction(request, response, mapping);

这个dispatcher就是org.apache.struts2.dispatcher.Dispatcher类,即执行Dispatcher类的serviceAction方法,继续跟进去。这个方法为:

public void serviceAction(HttpServletRequest request, HttpServletResponse response, ActionMapping mapping)
            throws ServletException {

        Map<String, Object> extraContext = createContextMap(request, response, mapping);

        ValueStack stack = (ValueStack) request.getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
      ......
        try {
            UtilTimerStack.push(timerKey);
            String namespace = mapping.getNamespace();
            String name = mapping.getName();
            String method = mapping.getMethod();

            ActionProxy proxy = getContainer().getInstance(ActionProxyFactory.class).createActionProxy(
                    namespace, name, method, extraContext, true, false);

            request.setAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY, proxy.getInvocation().getStack());

            if (mapping.getResult() != null) {
                Result result = mapping.getResult();
                result.execute(proxy.getInvocation());
            } else {
                proxy.execute();
            }
         ......
    }

说明:

  • 1.这个方法中我们首先看这两行
ValueStack stack = (ValueStack) request.getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
request.setAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY, proxy.getInvocation().getStack());

可以看到struts2将值栈存到了request域中,我们以后要想取得值栈中的内容可以使用上面的常量。

  • 2.同时可以看到产生了一个ActionProxy对象,这个对象会通过ConfigurationManager去读取配置文件的相关信息,之后就会执行其execute()方法。跟踪此方法会发现此方法最后
return invocation.invoke();

也就是执行com.opensymphony.xwork2.DefaultActionInvocation对象的invoke方法,而此类中我们可以看到包含了我们自己的action和一系列的拦截器(interceptor)。跟踪此方法:

if (interceptors.hasNext()) {
                final InterceptorMapping interceptor = interceptors.next();
                String interceptorMsg = "interceptor: " + interceptor.getName();
                UtilTimerStack.push(interceptorMsg);
                try {
                                resultCode = interceptor.getInterceptor().intercept(DefaultActionInvocation.this);
                            }
                finally {
                    UtilTimerStack.pop(interceptorMsg);
                }
            } else {
                resultCode = invokeActionOnly();
            }

说明:代码中这样一段可以看到首先执行各类拦截器,当拦截器执行完之后执行invokeAction()这个方法。
首先我们看

resultCode = interceptor.getInterceptor().intercept(DefaultActionInvocation.this);

这里调用的是拦截器的intercept方法,在此方法中我们可以看到

result = invocation.invoke();

也就是说继续调用DefaultActionInvocationinvoke方法,这样就会调用下一个拦截器,当将所有拦截器都调用一遍之后,才会执行DefaultActionInvocationinvokeActionOnly();方法,我们跟进去会发现执行的是

return invokeAction(getAction(), proxy.getConfig());

invokeAction方法中:

String methodName = proxy.getMethod();

此时拿到的方法就是我们的action中的execute方法,然后就执行此方法,之后就返回去,经过一些列的拦截器之后返回到
com.opensymphony.xwork2.DefaultActionInvocation然后返回到org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter类,最后在页面中响应。

最后:根据原理图和debug调试基本上可以对struts的原理有个比较大概的了解了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,319评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,801评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,567评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,156评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,019评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,090评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,500评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,192评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,474评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,566评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,338评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,212评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,572评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,890评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,169评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,478评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,661评论 2 335

推荐阅读更多精彩内容