`

难经3:Struts2,拦截器拦不住Result?

阅读更多

[问题]

使用Struts2作为web框架,知道它的拦截器(Interceptor)机制,类似与Filter和Spring的AOP,于是实现了一个为Action增加自定义前置(before)动作和后置动作(after)的拦截器(曰:WInterceptor),不过用一段时间发现,在WInterceptor的after中,对Action对象的属性修改在页面看不到,对请求对象的属性设置也无效。为什么在调用了Action之后(invokeAction())之后,request就不能使用了呢,拦截器不能改变Action的Result么?

 

[探幽]

 

在重看了Struts2的拦截器的官方文档以后,还是不明白上面的问题是为什么。地球人都知道,Struts2其实就是Webwork2,而拦截器的核心实现在XWork,利用XWork的拦截器框架,Struts2在外围通过线程上下文,绑定了Request和Response对象的包装类,哪问题到底在Struts2,还是在XWork?

 

在看到下面这张调用图,我才突然反应过来,“我真笨,真的,我只知道拦截器调用栈的最底层,是Action方法的调用,却不知道Result的调用也是在栈底调用,之后才返回给上一个拦截器,层层退出”:



        感谢这张图的作者,它简单,但有效。

 

 问题的关键在于,在调用actionInvocation.invoke()的之后,不仅执行类Action,也执行类Result。因而,等退回到拦截器的调用代码时,Result已经生成,View已经确定,这时你再修改模型(Action的属性)或请求对象的属性,对视图不会有任何影响。

 

另,为什么Result的执行不放到拦截器链的外面呢?这是我开始的直觉,有知道的朋友烦告知一声。

[解难]

 方法一:使用现成的PreResultListener监听器事件

搞清楚原因,卷起袖子干吧,只要让WInterpretor的after事件,放在Result的生成之前就行了。

看看XWork的拦截器接口注入的actionInvocation,其实就提供增加Result执行的前置监听事件-PreResultListener:

    /**
     * Register a {@link PreResultListener} to be notified after the Action is executed and
     * before the Result is executed.
     * <p/>
     * The ActionInvocation implementation must guarantee that listeners will be called in
     * the order in which they are registered.
     * <p/>
     * Listener registration and execution does not need to be thread-safe.
     *
     * @param listener the listener to add.
     */
    void addPreResultListener(PreResultListener listener);

 

因此,让拦截器实现这个接口,就可以自然实现Action执行after事件了。

 

 

方法二,实现自己的 ActionInvocation ,手动分离Action和Result的执行

本来前面的方法已经很好了,可是,可是啊,在addPreResultListener里的异常,不会被Struts的框架捕获,而且,addPreResultListener接口不能传递自己的上下文参数,难道动用ThreadLocal传参?

 

研究了一下XWork的ActionInvocation 接口默认实现类DefaultActionInvocation, 写了一个包装类,将Action的执行和Result的生成完全分开,或许有人用的着,放上来,见附件(ActionInvocationWrapper),如有不妥之处请告知。

 

exeucteAction是执行Action,executeResult是执行Result

 

  • 大小: 17.5 KB
5
0
分享到:
评论
14 楼 yanglphf 2014-01-17  
长知识了,谢谢
13 楼 buddie 2011-11-01  
为什么,执行Action和生成Result分开后,生成的页面中内容会出现两次。就是一个页面中,页面内容在上面显示完后,下面又重复显示了一次?
12 楼 buddie 2011-10-31  
非常感谢ixu的这篇文章,让我成功的解决我所遇到了问题,并让我有了想去看Struts2源码的想法(对于我这种菜鸟来说,过去从没想过!)。
11 楼 yudylaw 2009-04-15  
看了你的两篇 文章,收获不少。
10 楼 downpour 2009-02-16  
ixu 写道

在“[url=http://liuu.iteye.com/blog/333317]难经4:Struts2,拦截器拦不住的异常?![url]”中,我对这个问题做了更清晰详尽的分析和描述,也有了更好的解决办法。


我看了一下,暂时好像没看出漏洞来,应该是ActionInvocation的代码的逻辑问题。我还需要搭一个环境试试看这个问题。

不过实际上,这样的话,拦截器还是能够拦截到exception,ActionInvocation所控制的执行顺序出了问题。

可能是我从来不用struts2自带的Exception的拦截器,我基本上会自己写一个用于拦截自己的BusinessException,并做一些业务判断,所以也用不上<global-exception-mappings>的配置。

谢谢你的分析,让我学到很多东西。
9 楼 ixu 2009-02-16  
在“[url=http://liuu.iteye.com/blog/333317]难经4:Struts2,拦截器拦不住的异常?![url]”中,我对这个问题做了更清晰详尽的分析和描述,也有了更好的解决办法。
8 楼 downpour 2009-01-17  
如果我的回复还不能足以让你相信的话,请看一下struts2的reference:

thisWillRunFirstInterceptor
  thisWillRunNextInterceptor
    followedByThisInterceptor
      thisWillRunLastInterceptor
        MyAction1
        MyAction2 (chain)
        MyPreResultListener
        MyResult (result)
      thisWillRunLastInterceptor
    followedByThisInterceptor
  thisWillRunNextInterceptor
thisWillRunFirstInterceptor

如果你的thisWillRunFirstInterceptor写了一个try catch,你认为MyPreResultListener不会被捕获到?
7 楼 downpour 2009-01-17  
你的态度过激了,我不和你继续讨论下去了。

我所说的所有的东西全部都有代码的支持。我从Webwork开始一直到Struts2已经用了快4年了,从来没有听说过PreResultListener的异常不能被上层的拦截器拦截。在我自己编写的拦截器中,我也不止一次的抛出过RuntimeException,从来没有出现过异常无法被拦截。所以按照我的经验和实践,你的结论是错的。希望你仔细验证你的代码的问题。
6 楼 ixu 2009-01-16  
如果你想继续探讨这个问题,请:
1、真正写一个拦截器,实现PreResultListener
2、  A.在intercept方法中直接抛出异常(RuntimeException就行)
或者 B.在beforeResult方法中抛出异常
这两个可以用请求参数控制
3、写一个TestAction配上默认拦截器栈和这个拦截器,再配上一个全局的异常映射
4、在Servlet服务器(如Tomcat)下运行,访问TestAction

请观察2.A和2.B的响应结果的不同...

我说自己没有仔细去看源码,请正确理解“仔细”的含义,正因为看过源码,才奇怪在PreResultListener的异常应该是被上层拦截器捕捉的,但是最终却导致了Dispatcher抛出ServletException,对于这一点,我还没有“仔细”看,因而“不清楚原因”。这并不影响我说的现象和结论,在研究代码行为时,最好先把源码当黑箱,不要对源码想当然。

如果你宁愿一直看着源码,做一天的争论,也不愿拿十分钟来写一个测试验证你的猜疑,那么我可以严重怀疑你真的看清了、看懂了源码么?

还是那句话,实践是检验真理的唯一标准。

另,对于参数,确实有很多你想象不到的场景,你说的用ActionContext,其实不就是被包装了的线程上下文么?!
5 楼 downpour 2009-01-16  
所以说嘛,我最近正好在写Struts2的文章,至于Dispatcher,DefaultActionInvocation这些关键类更是读了好多遍,按照我的理解,是不可能发生在PreResultListener里抛出异常,而拦截器无法拦截到的情况的。因为DefaultActionInvocation中的invoke是递归调用,所以在上层的拦截器会包住整个执行过程。所以,既然你还没有研究过源码,不妨去看一下,我对你的这个结论是表示怀疑的。

至于第二个问题,我比较困惑的是,你究竟需要传递什么参数进去?

public void beforeResult(ActionInvocation invocation, String resultCode)


在这个接口中有ActionInvocation,而在ActionInvocation中,我们可以获取ActionContext,从而可以获取一切Action执行过程中的数据。

如果你需要传递额外的参数(恕我实在无法想到这种场景的存在性),你还可以通过ActionInvocation.getInvocationContext().put方法把变量放到ActionContext中进行传递。这个线程总是安全了的吧?
4 楼 ixu 2009-01-16  
downpour 同学,第一个问题已经说了,你完全可以自测试一下嘛,至于源码,你可以去看Struts2的Dispaatcher、XWork的ExceptionMappingInterceptor还有DefaultActionInvocation的代码,不用我在这里贴出来吧。至于为什么会这样,我也还没有仔细看源码,所有暂时还不是很清楚,你如果找出原因了,麻烦告知一下。

如果要用拦截器同时实现PreResultListener接口,第二个问题确实存在,因为拦截器一般都是单例,不能把在拦截过程中用到的数据用属性保存,否则线程不安全,所有不能直接通过属性字段传东西给实现PreResultListener接口的方法,而这个方法的参数又是固定的,因此或许只有用线程上下文了,不知我说明白没有。
3 楼 downpour 2009-01-16  
关于第一个问题,我还是深表怀疑,请用源码证明一下。

关于第二个问题,我认为你的理解也有问题,不用匿名类就不能传参数到PreResultListener的实现类了嘛?也请贴出代码。
2 楼 ixu 2009-01-16  
引用
1. addPreResultListener里的异常,不会被Struts的框架捕获

这个结论怎么得到的?


在使用全局异常映射配置<global-exception-mappings>时,如果Action执行有异常的话,会跳转到指定的异常结果页面。不过很遗憾,如果你在PreResultListener里抛出异常,Struts2的Dispatcher类会直接当成ServletException抛出给Servlet容器,去不到全局异常页面。你可以试一下。

引用

2. addPreResultListener接口不能传递自己的上下文参数,难道动用ThreadLocal传参

什么叫自己的上下文参数?不太明白


其实就是想传自己的参数给接口的实现代码,其实这个好解决,不用拦截器实现接口,直接用匿名内部类就好了,但是第一个问题不好办。

1 楼 downpour 2009-01-16  
引用
在addPreResultListener里的异常,不会被Struts的框架捕获,而且,addPreResultListener接口不能传递自己的上下文参数,难道动用ThreadLocal传参?


2点疑问。

1. addPreResultListener里的异常,不会被Struts的框架捕获

这个结论怎么得到的?

2. addPreResultListener接口不能传递自己的上下文参数,难道动用ThreadLocal传参

什么叫自己的上下文参数?不太明白

相关推荐

    经络学说中存在的瑕疵与疑惑

    经络学说中存在的瑕疵与疑惑,王维兵,王玉玲,目的:分析经络学说中存在的瑕疵与疑惑,为修正经络学说奠定基础。方法:以《内经》、《难经》、高校教材《经络学》等为依据,与

    “脾裹血”功能与实质的现代释义

    “脾裹血”功能与实质的现代释义,刘雅峰,王佳,经考证历史文献,结合临床实践认为,《难经》中提出的

    基于matlab实现实现了基于项目的协同过滤代码,MATLAB实现.rar

    基于matlab实现实现了基于项目的协同过滤代码,MATLAB实现.rar

    各地区年末城镇登记失业人员及失业率.xls

    数据来源:中国劳动统计NJ-2023版

    企业固定资产信息管理系统设计与实现.doc

    企业固定资产信息管理系统设计与实现.doc

    node-v11.14.0-darwin-x64.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    node-v8.9.1-sunos-x64.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    node-v12.10.0-linux-armv7l.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    基于VB实现的学生成绩管理系统(源代码+系统+开题报告+答辩PPT).zip

    【作品名称】:基于VB实现的学生成绩管理系统(源代码+系统+开题报告+答辩PPT) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    银行信贷管理系统设计与实现-(毕业设计)1.docx

    银行信贷管理系统设计与实现-(毕业设计)1.docx

    基于VB实现的银行代扣代发工资系统(源代码+系统+开题报告).zip

    【作品名称】:基于VB实现的银行代扣代发工资系统(源代码+系统+开题报告) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    (更新至2022年)全国镇分年龄、性别的人口数.xls

    数据来源:中国人口与就业统计NJ-2023版

    基于Java的考试管理系统

    java,大学课后作业

    数据更新至2020年分地区发电装机容量增速(风电).xls

    数据来源:中国电力统计NJ-2021版

    基于VB实现的网上餐饮管理系统设计(论文+源代码+开题报告+英文文献).zip

    【作品名称】:基于VB实现的网上餐饮管理系统设计(论文+源代码+开题报告+英文文献) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    基于matlab实现烟花算法进行函数最小-最大值寻优,对十多个测试函数进行了寻优,效果良好,代码附带详细说明.rar

    基于matlab实现烟花算法进行函数最小_最大值寻优,对十多个测试函数进行了寻优,效果良好,代码附带详细说明.rar

    数据更新至2020年分地区单机6000千瓦及以上 水力发电机组分类情况(合计).xls

    数据来源:中国电力统计NJ-2021版

    基于VB实现的商场管理系统设计(源代码+系统).zip

    【作品名称】:基于VB实现的商场管理系统设计(源代码+系统) 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。

    (更新至2022年)全国按年龄、性别分的就业人员受教育程度构成.xls

    数据来源:中国劳动统计NJ-2023版

Global site tag (gtag.js) - Google Analytics