struts2源码分析笔记

来源:互联网 发布:安卓开发php服务器端 编辑:程序博客网 时间:2024/05/25 08:12

这是Struts2官方站点提供的Struts 2 的整体结构。


Struts2请求响应流程:


struts2工作原理

   1 客户端发送请求;
   2 这个请求经过一系列的过滤器(Filter)(这些过滤器中有一个叫做ActionContextCleanUp的可选过滤器,这个过滤器对于Struts2和其他框架的集成很有帮助,例如:SiteMesh Plugin)

   3 接着FilterDispatcher被调用,FilterDispatcher询问ActionMapper来决定这个请求是否需要调用某个Action。

       FilterDispatcher的功能如下:

            (1)执行Actions
            (2)清除ActionContext
            (3)维护静态内容
            (4)清除request生命周期内的XWork的interceptors
   4 如果ActionMapper决定需要调用某个Action,FilterDispatcher把请求的处理交给ActionProxy
   5 ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类
   6 ActionProxy创建一个ActionInvocation的实例。
   7 ActionInvocation实例使用命名模式来调用,在调用Action的过程前后,涉及到相关拦截器(Intercepter)的调用。
   8 一旦Action执行完毕,ActionInvocation负责根据struts.xml中的配置找到对应的返回结果。返回结果通常是(但不总是,也可 能是另外的一个Action链)一个需要被表示的JSP或者FreeMarker的模版。在表示的过程中可以使用Struts2 框架中继承的标签。在这个过程中需要涉及到ActionMapper

   action配置例子:
   <package name="login" extends="baseAction" namespace="/">
<action name="login_*" class="loginAction" method="{1}">
<result name="userLoginSuccess">/res/templet/user/templet/new/index.jsp</result>
<result name="userLoginFail">/res/templet/user/templet/new/login.jsp</result>
<result name="userLoginExitSuccess">/res/templet/user/templet/new/login.jsp</result>
</action>
   </package>

   拦截器配置例子:
   <interceptors>
        <interceptor name="userActionInter" class="com.tfl.shop.common.UserActionInterceptor"></interceptor>
       
        <interceptor-stack name="userActionInterStack">
        <interceptor-ref name="userActionInter"></interceptor-ref>
        <interceptor-ref name="defaultStack"></interceptor-ref>
        </interceptor-stack>
       
   </interceptors>
----------------------
--------
web.xml可以配FilterDispatcher(从Struts 2.1.3开始,它已不推荐使用)和StrutsPrepareAndExecuteFilter这两个过滤器
FilterDispatcher是struts2.0.x到2.1.2版本的核心过滤器! StrutsPrepareAndExecuteFilter是自2.1.3开始就替代了FilterDispatcher的!
--------

默认解析的.xml文件
[struts-default.xml, struts-plugin.xml, struts.xml],如果不想使用默认名称的xml文件,可以直接在web.xml中配置config初始化参数即可。

--------
看了源代码后初步总结流程:
请求-->StrutsPrepareAndExecuteFilter.doFilter方法-->PrepareOperations.findActionMapping方法得到ActionMapping(此类存放action名、命名空间、方法名等)-->Dispatcher.serviceAction方法-->ConfigurationManager.getConfiguration方法-->创建DefaultActionProxyFactory对象-->StrutsActionProxyFactory.createActionProxy方法创建StrutsActionProxy对象-->StrutsActionProxy.execute方法-->DefaultActionInvocation.invoke方法-->递归执行所有拦截器类-->DefaultActionInvocation.invokeAction方法-->自定义的Action类中的方法-->PreResultListener(执行Result操作前的监听器)-->执行Result-->递归执行所有拦截器类-->根据配置文件返回相应的view

请求-->FilterDispatcher.doFilter方法-->ActionMapper.getMapping方法ActionMapping(此类存放action名、命名空间、方法名等)-->Dispatcher.serviceAction方法-->ConfigurationManager.getConfiguration方法-->创建DefaultActionProxyFactory对象-->StrutsActionProxyFactory.createActionProxy方法创建StrutsActionProxy对象-->StrutsActionProxy.execute方法-->DefaultActionInvocation.invoke方法-->递归执行所有拦截器类-->DefaultActionInvocation.invokeAction方法-->自定义的Action类中的方法-->PreResultListener(执行Result操作前的监听器)-->执行Result-->递归执行所有拦截器类-->根据配置文件返回相应的view

注意:
1. 流程提到的Default开头都是实现类,例如:DefaultActionProxyFactory是ActionProxyFactory(接口类)实现类;
2. StrutsActionProxy类继承DefaultActionProxy类,StrutsActionProxyFactory类继承DefaultActionProxyFactory类;
3. DefaultActionProxyFactory创建StrutsActionProxy对象而不创建DefaultActionProxy对象,整个流程不调用DefaultActionProxy.execute方法,而是调用       StrutsActionProxy.execute方法,StrutsActionProxy.execute方法中调用DefaultActionInvocation.invoke方法。
--------
拦截器可以在action方法执行前后执行一些逻辑
拦截器执行顺序:
第一个拦截器action前逻辑
第二个拦截器action前逻辑
action方法
第二个拦截器action后逻辑
第一个拦截器action后逻辑

--------

ActionMapper
ActionMapper其实是HttpServletRequest和Action调用请求的一个映射,它屏蔽了Action对于Request等javaServlet类的依赖。Struts2中它的默认实现类是DefaultActionMapper,ActionMapper很大的用处可以根据自己的需要来设计url格式,它自己也有Restful的实现,具体可以参考文档的docs\actionmapper.html。

ActionProxy&ActionInvocation
Action的一个代理,由ActionProxyFactory创建,它本身不包括Action实例,默认实现DefaultActionProxy是由ActionInvocation持有Action实例。ActionProxy作用是如何取得Action,无论是本地还是远程。而ActionInvocation的作用是如何执行Action,拦截器的功能就是在ActionInvocation中实现的。

ConfigurationProvider&Configuration

ConfigurationProvider就是Struts2中配置文件的解析器,Struts2中的配置文件主要是尤其实现类XmlConfigurationProvider及其子类StrutsXmlConfigurationProvider来解析。


Struts2 是一个相当强大的Java Web开源框架,是一个基于POJO的Action的MVC Web框架。它基于当年的Webwork和XWork框架,继承其优点,同时做了相当的改进。Struts2现在在Java Web开发界的地位可以说是大红大紫,从开发人员的角度来分析,Struts2之所以能够如此的深入开发人员之心,与其优良的设计是分不开的。
 
struts2的优点如下:
1.Struts2基于MVC架构,框架结构清晰,开发流程一目了然,开发人员可以很好的掌控开发的过程。 
我在项目开发过程中,一个具体的功能的开发流程是:拿到一个具体的功能需求文档和设计好的前台界面(在开发中我不负责设计页面),分析需要从前台传递哪些参数,确定参数的变量名称,在Action中设置相应的变量,这些参数在前台如何显示,并将页面上的一些控件适当使用Struts2提供的服务器端控件来代替,编写Action对应的方法来完成业务逻辑,最后,做一些与配置文件相关的设置。当然实际的开发比这个过程要复杂,涉及到数据库,验证,异常等处理。但是使用Struts2进行开发,你的关注点绝大部分是在如何实现业务逻辑上,开发过程十分清晰明了。 

2.使用OGNL进行参数传递。 
OGNL提供了在Struts2里访问各种作用域中的数据的简单方式,你可以方便的获取Request,Attribute,Application,Session,Parameters中的数据。大大简化了开发人员在获取这些数据时的代码量。 

3.强大的拦截器 
Struts2 的拦截器是一个Action级别的AOP,Struts2中的许多特性都是通过拦截器来实现的,例如异常处理,文件上传,验证等。拦截器是可配置与重用的,可以将一些通用的功能如:登录验证,权限验证等置于拦截器中以完成一些Java Web项目中比较通用的功能。在我实现的的一Web项目中,就是使用Struts2的拦截器来完成了系统中的权限验证功能。 

4.易于测试 
Struts2的Action都是简单的POJO,这样可以方便的对Struts2的Action编写测试用例,大大方便了Java Web项目的测试。 

5.易于扩展的插件机制 
在Struts2添加扩展是一件愉快而轻松的事情,只需要将所需要的Jar包放到WEB-INF/lib文件夹中,在struts.xml中作一些简单的设置就可以实现扩展。常用的Struts2的扩展可以通过这个链接找到: http://cwiki.apache.org/S2PLUGINS/home.html 

6.模块化 
Struts2已经把模块化作为了体系架构中的基本思想,可以通过三种方法来将应用程序模块化: 
将配置信息拆分成多个文件 
把自包含的应用模块创建为插件 
创建新的框架特性,即将与特定应用无关的新功能组织成插件,以添加到多个应用中去。 

7.全局结果与声明式异常 
为应用程序添加全局的Result,和在配置文件中对异常进行处理,这样当处理过程中出现指定异常时,可以跳转到特定页面,这一功能十分实用。 

1 0
原创粉丝点击