首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 其他教程 > 开源软件 >

struts2请求历程源码分析

2013-11-08 
struts2请求过程源码分析  Struts2框架的正常运行,除了占核心地位的xwork的支持以外,Struts2本身也提供了

struts2请求过程源码分析

  Struts2框架的正常运行,除了占核心地位的xwork的支持以外,Struts2本身也提供了许多类,这些类被分门别类组织到不同的包中。从源代码中发现,基本上每一个Struts2类都访问了WebWork提供的功能,从而也可以看出Struts2与WebWork千丝万缕的联系。但无论如何,Struts2的核心功能比如将请求委托给哪个Action处理都是由xwork完成的,Struts2只是在WebWork的基础上做了适当的简化、加强和封装,并少量保留Struts1.x中的习惯。

以下是包说明:

org.apache.struts2. components

该包封装视图组件,Struts2在视图组件上有了很大加强,不仅增加了组件的属性个数,更新增了几个非常有用的组件,如updownselect、doubleselect、datetimepicker、token、tree等。

?另外,Struts2可视化视图组件开始支持主题(theme),缺省情况下,使用自带的缺省主题,如果要自定义页面效果,需要将组件的theme属性设置为simple。

org.apache.struts2. config该包定义与配置相关的接口和类。实际上,工程中的xml和properties文件的读取和解析都是由WebWork完成的,Struts只做了少量的工作。org.apache.struts2.dispatcherStruts2的核心包,最重要的类都放在该包中。org.apache.struts2.impl该包只定义了3个类,他们是StrutsActionProxy、StrutsActionProxyFactory、StrutsObjectFactory,这三个类都是对xwork的扩展。org.apache.struts2.interceptor定义内置的截拦器。org.apache.struts2.servlet用HttpServletRequest相关方法实现principalproxy接口。org.apache.struts2.util实用包。org.apache.struts2.views提供freemarker、jsp、velocity等不同类型的页面呈现。

?

?根目录下的5个文件说明:

StrutsStaticsStruts常数。常数可以用来获取或设置对象从行动中或其他集合。RequestUtils请求处理程序类。此类只有一个方法getServletPath,作用检索当前请求的servlet路径ServletActionContext网站的特定的上下文信息StrutsConstants该类提供了框架配置键的中心位置用于存储和检索配置设置。StrutsException通用运行时异常类

?

? ?

?

?

?

struts2 架构图如下图所示:

struts2请求历程源码分析

依照上图,我们可以看出一个请求在struts的处理大概有如下步骤:

  1、客户端初始化一个指向Servlet容器(例如Tomcat)的请求;

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

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

  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。

?

strut2源码分析:

  首先我们使用struts2框架都会在web.xml中注册和映射struts2,配置内容如下:

1 <filter> 2     <filter-name>struts2</filter-name>  3     <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> 4   </filter>  5   <filter-mapping> 6     <filter-name>struts2</filter-name>  7     <url-pattern>/*</url-pattern> 8   </filter-mapping>

注:在早期的struts2中,都是使用FilterDispathcer,从Struts 2.1.3开始,它已不推荐使用。如果你使用的Struts的版本 >= 2.1.3,推荐升级到新的Filter,StrutsPrepareAndExecuteFilter。在此研究的是StrutsPrepareAndExecuteFilter。

  StrutsPrepareAndExecuteFilter中的方法:

void init(FilterConfig filterConfig)?继承自Filter,过滤器的初始化doFilter(ServletRequest req, ServletResponse res, FilterChain chain)?继承自Filter,执行过滤器void destroy()继承自Filter,用于资源释放void postInit(Dispatcher dispatcher, FilterConfig filterConfig)?Callback for post initialization(一个空的方法,用于方法回调初始化)

?web容器一启动,就会初始化核心过滤器StrutsPrepareAndExecuteFilter,并执行初始化方法,初始化方法如下:

 1 public void init(FilterConfig filterConfig) throws ServletException { 2         InitOperations init = new InitOperations(); 3         Dispatcher dispatcher = null; 4         try { 5             //封装filterConfig,其中有个主要方法getInitParameterNames将参数名字以String格式存储在List中 6             FilterHostConfig config = new FilterHostConfig(filterConfig); 7             //初始化struts内部日志 8             init.initLogging(config); 9             //创建dispatcher ,并初始化10             dispatcher = init.initDispatcher(config);11             init.initStaticContentLoader(config, dispatcher);12             //初始化类属性:prepare 、execute13             prepare = new PrepareOperations(filterConfig.getServletContext(), dispatcher);14             execute = new ExecuteOperations(filterConfig.getServletContext(), dispatcher);15             this.excludedPatterns = init.buildExcludedPatternsList(dispatcher);16             //回调空的postInit方法17             postInit(dispatcher, filterConfig);18         } finally {19             if (dispatcher != null) {20                 dispatcher.cleanUpAfterInit();21             }22             init.cleanup();23         }24     }

?关于封装filterConfig,首先看下FilterHostConfig ,源码如下:

 1 public class FilterHostConfig implements HostConfig { 2  3     private FilterConfig config; 4     //构造方法 5     public FilterHostConfig(FilterConfig config) { 6         this.config = config; 7     } 8     //根据init-param配置的param-name获取param-value的值   9     public String getInitParameter(String key) {10         return config.getInitParameter(key);11     }12     //返回初始化参数名的迭代器 13     public Iterator<String> getInitParameterNames() {14         return MakeIterator.convert(config.getInitParameterNames());15     }16     //返回Servlet上下文17     public ServletContext getServletContext() {18         return config.getServletContext();19     }20 }

  只有短短的几行代码,getInitParameterNames是这个类的核心,将Filter初始化参数名称有枚举类型转为Iterator。此类的主要作为是对filterConfig 封装。

  接下来,看下StrutsPrepareAndExecuteFilter中init方法中dispatcher = init.initDispatcher(config);这是初始化dispatcher的,是通过init对象的initDispatcher方法来初始化的,init是InitOperations类的对象,我们看看InitOperations中initDispatcher方法:

 1 private Dispatcher createDispatcher( HostConfig filterConfig ) { 2         //存放参数的Map 3         Map<String, String> params = new HashMap<String, String>(); 4         //将参数存放到Map 5         for ( Iterator e = filterConfig.getInitParameterNames(); e.hasNext(); ) { 6             String name = (String) e.next(); 7             String value = filterConfig.getInitParameter(name); 8             params.put(name, value); 9         }10         //根据servlet上下文和参数Map构造Dispatcher 11         return new Dispatcher(filterConfig.getServletContext(), params);12     }

  这样dispatcher对象创建完成,接着就是dispatcher对象的初始化,打开Dispatcher类,看到它的init方法如下:

 1 public void init() { 2  3         if (configurationManager == null) { 4             configurationManager = createConfigurationManager(BeanSelectionProvider.DEFAULT_BEAN_NAME); 5         } 6  7         try { 8             init_FileManager(); 9             //加载org/apache/struts2/default.properties10             init_DefaultProperties(); // [1]11             //加载struts-default.xml,struts-plugin.xml,struts.xml12             init_TraditionalXmlConfigurations(); // [2]13             init_LegacyStrutsProperties(); // [3]14             //用户自己实现的ConfigurationProviders类 15             init_CustomConfigurationProviders(); // [5]16             //Filter的初始化参数 17             init_FilterInitParameters() ; // [6]18             init_AliasStandardObjects() ; // [7]19 20             Container container = init_PreloadConfiguration();21             container.inject(this);22             init_CheckWebLogicWorkaround(container);23 24             if (!dispatcherListeners.isEmpty()) {25                 for (DispatcherListener l : dispatcherListeners) {26                     l.dispatcherInitialized(this);27                 }28             }29         } catch (Exception ex) {30             if (LOG.isErrorEnabled())31                 LOG.error("Dispatcher initialization failed", ex);32             throw new StrutsException(ex);33         }34     }

  这里主要是加载一些配置文件的,将按照顺序逐一加载:default.properties,struts-default.xml,struts-plugin.xml,struts.xml,……关于文件是如何加载的,大家可以自己取看源文件,主要是由xwork核心类加载的,代码在xwork-core\src\main\java\com\opensymphony\xwork2\config\providers包里面。

  现在,我们回到StrutsPrepareAndExecuteFilter类中,刚才我们分析了StrutsPrepareAndExecuteFilter类的init方法,该方法在web容器一启动就会调用的,当用户访问某个action的时候,首先调用核心过滤器StrutsPrepareAndExecuteFilter的doFilter方法,该方法内容如下:

 1 public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { 2  3         HttpServletRequest request = (HttpServletRequest) req; 4         HttpServletResponse response = (HttpServletResponse) res; 5  6         try { 7             //设置编码和国际化 8             prepare.setEncodingAndLocale(request, response); 9             //创建action上下文10             prepare.createActionContext(request, response);11             prepare.assignDispatcherToThread();12             if (excludedPatterns != null && prepare.isUrlExcluded(request, excludedPatterns)) {13                 chain.doFilter(request, response);14             } else {15                 request = prepare.wrapRequest(request);16                 ActionMapping mapping = prepare.findActionMapping(request, response, true);17                 //如果mapping为空,则认为不是调用action,会调用下一个过滤器链,直到获取到mapping才调用action18                 if (mapping == null) {19                     boolean handled = execute.executeStaticResourceRequest(request, response);20                     if (!handled) {21                         chain.doFilter(request, response);22                     }23                 } else {24                     //执行action25                     execute.executeAction(request, response, mapping);26                 }27             }28         } finally {29             prepare.cleanupRequest(request);30         }31     }

  下面对doFilter方法中的重点部分一一讲解:

(1)prepare.setEncodingAndLocale(request, response);

  第8行是调用prepare对象的setEncodingAndLocale方法,prepare是PrepareOperations类的对象,PrepareOperations类是用来做请求准备工作的。我们看下PrepareOperations类中的setEncodingAndLocale方法:

 1 public void prepare(HttpServletRequest request, HttpServletResponse response) { 2         String encoding = null; 3         if (defaultEncoding != null) { 4             encoding = defaultEncoding; 5         } 6         // check for Ajax request to use UTF-8 encoding strictly http://www.w3.org/TR/XMLHttpRequest/#the-send-method 7         if ("XMLHttpRequest".equals(request.getHeader("X-Requested-With"))) { 8             encoding = "UTF-8"; 9         }10 11         Locale locale = null;12         if (defaultLocale != null) {13             locale = LocalizedTextUtil.localeFromString(defaultLocale, request.getLocale());14         }15 16         if (encoding != null) {17             applyEncoding(request, encoding);18         }19 20         if (locale != null) {21             response.setLocale(locale);22         }23 24         if (paramsWorkaroundEnabled) {25             request.getParameter("foo"); // simply read any parameter (existing or not) to "prime" the request26         }27     }

  我们可以看到该方法只是简单的设置了encoding 和locale ,做的只是一些辅助的工作。

(2)prepare.createActionContext(request, response)

  我们回到StrutsPrepareAndExecuteFilter的doFilter方法,看到第10行代码:prepare.createActionContext(request, response);这是action上下文的创建,ActionContext是一个容器,这个容易主要存储request、session、application、parameters等相关信 息.ActionContext是一个线程的本地变量,这意味着不同的action之间不会共享ActionContext,所以也不用考虑线程安全问 题。其实质是一个Map,key是标示request、session、……的字符串,值是其对应的对象,我们可以看到com.opensymphony.xwork2.ActionContext类中时如下定义的:

 1 /** 2      * Creates the action context and initializes the thread local 3      */ 4     public ActionContext createActionContext(HttpServletRequest request, HttpServletResponse response) { 5         ActionContext ctx; 6         Integer counter = 1; 7         Integer oldCounter = (Integer) request.getAttribute(CLEANUP_RECURSION_COUNTER); 8         if (oldCounter != null) { 9             counter = oldCounter + 1;10         }11         //此处是从ThreadLocal中获取此ActionContext变量12         ActionContext oldContext = ActionContext.getContext();13         if (oldContext != null) {14             // detected existing context, so we are probably in a forward15             ctx = new ActionContext(new HashMap<String, Object>(oldContext.getContextMap()));16         } else {17             ValueStack stack = dispatcher.getContainer().getInstance(ValueStackFactory.class).createValueStack();18             stack.getContext().putAll(dispatcher.createContextMap(request, response, null, servletContext));19             //stack.getContext()返回的是一个Map<String,Object>,根据此Map构造一个ActionContext20             ctx = new ActionContext(stack.getContext());21         }22         request.setAttribute(CLEANUP_RECURSION_COUNTER, counter);23         //将ActionContext存到ThreadLocal 24         ActionContext.setContext(ctx);25         return ctx;26     }

?   上面第18行代码中dispatcher.createContextMap,如何封装相关参数:

 1 public Map<String,Object> createContextMap(HttpServletRequest request, HttpServletResponse response, 2             ActionMapping mapping, ServletContext context) { 3  4         // request map wrapping the http request objects 5         Map requestMap = new RequestMap(request); 6  7         // parameters map wrapping the http parameters.  ActionMapping parameters are now handled and applied separately 8         Map params = new HashMap(request.getParameterMap()); 9 10         // session map wrapping the http session11         Map session = new SessionMap(request);12 13         // application map wrapping the ServletContext14         Map application = new ApplicationMap(context);15         //requestMap、params、session等Map封装成为一个上下文Map 16         Map<String,Object> extraContext = createContextMap(requestMap, params, session, application, request, response, context);17 18         if (mapping != null) {19             extraContext.put(ServletActionContext.ACTION_MAPPING, mapping);20         }21         return extraContext;22     }

(3)request = prepare.wrapRequest(request)

  我们再次回到StrutsPrepareAndExecuteFilter的doFilter方法中,看到第15行:request = prepare.wrapRequest(request);这一句是对request进行包装的,我们看下prepare的wrapRequest方法:

 1 public HttpServletRequest wrapRequest(HttpServletRequest oldRequest) throws ServletException { 2         HttpServletRequest request = oldRequest; 3         try { 4             // Wrap request first, just in case it is multipart/form-data 5             // parameters might not be accessible through before encoding (ww-1278) 6             request = dispatcher.wrapRequest(request, servletContext); 7         } catch (IOException e) { 8             throw new ServletException("Could not wrap servlet request with MultipartRequestWrapper!", e); 9         }10         return request;11     }

  由第6行我们可以看到它里面调用的是dispatcher的wrapRequest方法,并且将servletContext对象也传进去了,我们看下dispatcher的wrapRequest:

 1 public HttpServletRequest wrapRequest(HttpServletRequest request, ServletContext servletContext) throws IOException { 2         // don't wrap more than once 3         if (request instanceof StrutsRequestWrapper) { 4             return request; 5         } 6  7         String content_type = request.getContentType(); 8         //如果content_type是multipart/form-data类型,则将request包装成MultiPartRequestWrapper对象,否则包装成StrutsRequestWrapper对象 9         if (content_type != null && content_type.contains("multipart/form-data")) {10             MultiPartRequest mpr = getMultiPartRequest();11             LocaleProvider provider = getContainer().getInstance(LocaleProvider.class);12             request = new MultiPartRequestWrapper(mpr, request, getSaveDir(servletContext), provider);13         } else {14             request = new StrutsRequestWrapper(request, disableRequestAttributeValueStackLookup);15         }16 17         return request;18     }

  此次包装根据请求内容的类型不同,返回不同的对象,如果为multipart/form-data类型,则返回MultiPartRequestWrapper类型的对象,该对象服务于文件上传,否则返回StrutsRequestWrapper类型的对象,MultiPartRequestWrapper是StrutsRequestWrapper的子类,而这两个类都是HttpServletRequest接口的实现。

(4)ActionMapping mapping = prepare.findActionMapping(request, response, true)

  包装request后,通过ActionMapper的getMapping()方法得到请求的Action,Action的配置信息存储在ActionMapping对象中,如StrutsPrepareAndExecuteFilter的doFilter方法中第16行:ActionMapping mapping = prepare.findActionMapping(request, response, true);我们找到prepare对象的findActionMapping方法:

 1 public ActionMapping findActionMapping(HttpServletRequest request, HttpServletResponse response, boolean forceLookup) { 2         //首先从request对象中取mapping对象,看是否存在 3         ActionMapping mapping = (ActionMapping) request.getAttribute(STRUTS_ACTION_MAPPING_KEY); 4         //不存在就创建一个 5         if (mapping == null || forceLookup) { 6             try { 7                 //首先创建ActionMapper对象,通过ActionMapper对象创建mapping对象 8                 mapping = dispatcher.getContainer().getInstance(ActionMapper.class).getMapping(request, dispatcher.getConfigurationManager()); 9                 if (mapping != null) {10                     request.setAttribute(STRUTS_ACTION_MAPPING_KEY, mapping);11                 }12             } catch (Exception ex) {13                 dispatcher.sendError(request, response, servletContext, HttpServletResponse.SC_INTERNAL_SERVER_ERROR, ex);14             }15         }16 17         return mapping;18     }

  下面是ActionMapper接口的实现类DefaultActionMapper的getMapping()方法的源代码:

 1 public ActionMapping getMapping(HttpServletRequest request, ConfigurationManager configManager) { 2         ActionMapping mapping = new ActionMapping(); 3         //获得请求的uri,即请求路径URL中工程名以后的部分,如/userAction.action 4         String uri = getUri(request); 5         //修正url的带;jsessionid 时找不到的bug 6         int indexOfSemicolon = uri.indexOf(";"); 7         uri = (indexOfSemicolon > -1) ? uri.substring(0, indexOfSemicolon) : uri; 8         //删除扩展名,如.action或者.do 9         uri = dropExtension(uri, mapping);10         if (uri == null) {11             return null;12         }13         //从uri中分离得到请求的action名、命名空间。14         parseNameAndNamespace(uri, mapping, configManager);15         //处理特殊的请求参数16         handleSpecialParameters(request, mapping);17         //如果允许动态方法调用,即形如/userAction!getAll.action的请求,分离action名和方法名18         return parseActionName(mapping);19     }

  下面对getMapping方法中的重要部分一一讲解:

  ①:parseNameAndNamespace(uri, mapping, configManager)

  我们主要看下第14行的parseNameAndNamespace(uri, mapping, configManager);这个方法的主要作用是分离出action名和命名空间:

 1 protected void parseNameAndNamespace(String uri, ActionMapping mapping, ConfigurationManager configManager) { 2         String namespace, name; 3         int lastSlash = uri.lastIndexOf("/"); //最后的斜杆的位置 4         if (lastSlash == -1) { 5             namespace = ""; 6             name = uri; 7         } else if (lastSlash == 0) { 8             // ww-1046, assume it is the root namespace, it will fallback to 9             // default10             // namespace anyway if not found in root namespace.11             namespace = "/";12             name = uri.substring(lastSlash + 1);13         //允许采用完整的命名空间,即设置命名空间是否必须进行精确匹配14         } else if (alwaysSelectFullNamespace) {15             // Simply select the namespace as everything before the last slash16             namespace = uri.substring(0, lastSlash);17             name = uri.substring(lastSlash + 1);18         } else {19             // Try to find the namespace in those defined, defaulting to ""20             Configuration config = configManager.getConfiguration();21             String prefix = uri.substring(0, lastSlash); //临时的命名空间,将会用来进行匹配22             namespace = "";//将命名空间暂时设为""23             boolean rootAvailable = false;//rootAvailable作用是判断配置文件中是否配置了命名空间"/"24             // Find the longest matching namespace, defaulting to the default25             for (Object cfg : config.getPackageConfigs().values()) { //循环遍历配置文件中的package标签26                 String ns = ((PackageConfig) cfg).getNamespace();    //获取每个package标签的namespace属性27                 //进行匹配28                 if (ns != null && prefix.startsWith(ns) && (prefix.length() == ns.length() || prefix.charAt(ns.length()) == '/')) {29                     if (ns.length() > namespace.length()) {30                         namespace = ns;31                     }32                 }33                 if ("/".equals(ns)) {34                     rootAvailable = true;35                 }36             }37 38             name = uri.substring(namespace.length() + 1);39 40             // Still none found, use root namespace if found41             if (rootAvailable && "".equals(namespace)) {42                 namespace = "/";43             }44         }45 46         if (!allowSlashesInActionNames) {47             int pos = name.lastIndexOf('/');48             if (pos > -1 && pos < name.length() - 1) {49                 name = name.substring(pos + 1);50             }51         }52         //将分离后的acion名和命名空间保存到mapping对象53         mapping.setNamespace(namespace);54         mapping.setName(cleanupActionName(name));55     }

  看到上面代码的第14行,参数alwaysSelectFullNamespace我们可以通过名字就能大概猜出来"允许采用完整的命名空间",即设置命名空间是否必须进行精确匹配,true必须,false可以模糊匹配,默认是false。进行精确匹配时要求请求url中的命名空间必须与配置文件中配置的某个命名空间必须相同,如果没有找到相同的则匹配失败。这个参数可通过struts2的"struts.mapper.alwaysSelectFullNamespace"常量配置,如:<constant name="struts.mapper.alwaysSelectFullNamespace" value="true" />。当alwaysSelectFullNamespace为true时,将uri以lastSlash为分割,左边的为namespace,右边的为name。如:http://localhost:8080/myproject/home/actionName!method.action,此时uri为/home/actionName!method.action(不过前面把后缀名去掉了,变成/home/actionName!method),lastSlash的,当前值是5,这样namespace为"/home", name为actionName!method。

  我们再看到上面代码第18行到第44行:当上面的所有条件都不满足时,其中包括alwaysSelectFullNamespace 为false(命名空间进行模糊匹配),将由此部分处理,进行模糊匹配。第1句,通过configManager.getConfiguration()从配置管理器中获得配置对象Configuration,Configuration中存放着struts2的所有配置,形式是将xml文档的相应元素封装为java bean,如<package>元素被封装到PackageConfig类中,这个一会儿会用到。第2句按lastSlash将uri截取出prefix,这是一个临时的命名空间,之后将会拿prefix进行模糊匹配。第3句namespace = "",将命名空间暂时设为""。第4句创建并设置rootAvailable,rootAvailable作用是判断配置文件中是否配置了命名空间"/",true为配置了,false未配置,下面for语句将会遍历我们配置的所有包(<package>),同时设置rootAvailable。第5句for,通过config.getPackageConfigs()获得所有已经配置的包,然后遍历。String ns = ((PackageConfig) cfg).getNamespace()获得当前包的命名空间ns,之后的if句是进行模糊匹配的核心,我摘出来单独说,如下:

 1 protected ActionMapping parseActionName(ActionMapping mapping) { 2         if (mapping.getName() == null) { 3             return null; 4         } 5         //如果允许动态方法调用 6         if (allowDynamicMethodCalls) { 7             // handle "name!method" convention. 8             String name = mapping.getName(); 9             int exclamation = name.lastIndexOf("!");10             //如果包含"!"就进行分离11             if (exclamation != -1) {12                 //分离出action名13                 mapping.setName(name.substring(0, exclamation));14                 //分离出方法名15                 mapping.setMethod(name.substring(exclamation + 1));16             }17         }18         return mapping;19     }

  到此为止getMapping方法已经分析结束了!

(5)execute.executeAction(request, response, mapping)

  上面我们分析完了mapping的获取,继续看doFilter方法:

 1 //如果mapping为空,则认为不是调用action,会调用下一个过滤器链,直到获取到mapping才调用action 2                 if (mapping == null) { 3                     boolean handled = execute.executeStaticResourceRequest(request, response); 4                     if (!handled) { 5                         chain.doFilter(request, response); 6                     } 7                 } else { 8                     //执行action 9                     execute.executeAction(request, response, mapping);10                 }

  如果mapping对象不为空,则会执行action,我们看到上面代码第9行:execute是ExecuteOperations类的对象,ExecuteOperations类在包org.apache.struts2.dispatcher.ng下面,我们找到它里面的executeAction方法:

 1 public void serviceAction(HttpServletRequest request, HttpServletResponse response, ServletContext context, 2                               ActionMapping mapping) throws ServletException { 3         //封转上下文环境,主要将requestMap、params、session等Map封装成为一个上下文Map 4         Map<String, Object> extraContext = createContextMap(request, response, mapping, context); 5  6         // If there was a previous value stack, then create a new copy and pass it in to be used by the new Action 7         ValueStack stack = (ValueStack) request.getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY); 8         boolean nullStack = stack == null; 9         if (nullStack) {10             ActionContext ctx = ActionContext.getContext();11             if (ctx != null) {12                 stack = ctx.getValueStack();13             }14         }15         if (stack != null) {16             extraContext.put(ActionContext.VALUE_STACK, valueStackFactory.createValueStack(stack));17         }18 19         String timerKey = "Handling request from Dispatcher";20         try {21             UtilTimerStack.push(timerKey);22             String namespace = mapping.getNamespace();//从mapping对象获取命名空间23             String name = mapping.getName();          //获取请求的action名24             String method = mapping.getMethod();      //获取请求方法25             //得到配置对象26             Configuration config = configurationManager.getConfiguration();27             //根据执行上下文参数,命名空间,名称等创建用户自定义Action的代理对象  28             ActionProxy proxy = config.getContainer().getInstance(ActionProxyFactory.class).createActionProxy(29                     namespace, name, method, extraContext, true, false);30 31             request.setAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY, proxy.getInvocation().getStack());32 33             // if the ActionMapping says to go straight to a result, do it!34             //如果配置文件中执行的这个action配置了result,就直接转到result35             if (mapping.getResult() != null) {36                 Result result = mapping.getResult();37                 result.execute(proxy.getInvocation());38             } else {39                 proxy.execute();40             }41 42             // If there was a previous value stack then set it back onto the request43             if (!nullStack) {44                 request.setAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY, stack);45             }46         } catch (ConfigurationException e) {47             // WW-2874 Only log error if in devMode48             if (devMode) {49                 String reqStr = request.getRequestURI();50                 if (request.getQueryString() != null) {51                     reqStr = reqStr + "?" + request.getQueryString();52                 }53                 LOG.error("Could not find action or result\n" + reqStr, e);54             } else {55                 if (LOG.isWarnEnabled()) {56                     LOG.warn("Could not find action or result", e);57                 }58             }59             sendError(request, response, context, HttpServletResponse.SC_NOT_FOUND, e);60         } catch (Exception e) {61             if (handleException || devMode) {62                 sendError(request, response, context, HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e);63             } else {64                 throw new ServletException(e);65             }66         } finally {67             UtilTimerStack.pop(timerKey);68         }69     }

  最后通过Result完成页面跳转!

?

总结:以前总是只会用struts2框架,对里面的原理没有一个很清晰的认识,这两天花时间把struts2框架的源码分析了一下,对它的工作原理有个更深的认识。既然是开源框架,有时间久得去研究研究它的源码,不然开源就失去了意义了,不要只停留在会用的层面上!

热点排行