struts2请求过程源码分析

1、struts2流程介绍

首先要知道struts2是在webwork的技术基础上开发的,采用拦截器的机制来处理用户请求的全新MVC框架。而webwork是建立在xwork的command模式框架之上的基于web的MVC框架。所以总而言之,无论是struts2还是webwork底层都是xwork

从其官方网站的介绍来看,XWork不仅提供了一系列基础构件,其中包括:一个IoC的容器、强大的表达式语言(OGNL)支持、数据类型转化、数据校验框架、可插拔的功能模块(插件模式)及其配置,并且在这一系列的基础构件之上,实现了一套基于Command设计模式的“事件请求执行框架”。
那么,XWork作为Struts2所依赖的底层核心,使得Struts2只需要关注与Web容器打交道的部分,而把其余的工作交给XWork即可。当Struts2收到一个Http请求时,Struts2只需要接收请求参数,交给XWork完成执行序列,当XWork执行完毕后,将结果交还Struts2返回相应的视图。

我这里的struts2源码是从官网下载的一个最新的struts-2.3.15.1-src.zip,将其解压即可。里面的目录页文件非常的多,我们只需要定位到struts-2.3.15.1\src\core\src\main\java\org\apache\struts2查看源文件。目录结构如下图:

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

struts2请求过程源码分析_第1张图片

根目录下的5个文件说明:
struts2请求过程源码分析_第2张图片

struts2 架构图如下图所示:

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

  1. 客户端初始化一个指向Servlet容器(例如Tomcat)的请求;
  2. 这个请求经过一系列的过滤器(Filter)(这些过滤器中有一个叫做ActionContextCleanUp的可选过滤器,这个过滤器对于Struts2和其他框架的集成很有帮助,例如:SiteMesh Plugin);
  3. 接着StrutsPrepareAndExecuteFilter被调用,执行doFilter()方法,询问ActionMapper来决定这个请求是否需要调用某个Action;
  4. 如果ActionMapper决定需要调用某个Action,StrutsPrepareAndExecuteFilter把请求的处理交给ActionProxy;
  5. ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类;
  6. ActionProxy创建一个ActionInvocation的实例。
  7. ActionInvocation实例使用命名模式来调用,在调用Action的过程前后,涉及到相关拦截器(Intercepter)的调用。
  8. 一旦Action执行完毕,ActionInvocation负责根据struts.xml中的配置找到对应的返回结果。返回结果通常是(但不总是,也可能是另外的一个Action链)一个需要被表示的JSP或者FreeMarker的模版。在表示的过程中可以使用Struts2 框架中继承的标签。在这个过程中需要涉及到ActionMapper。

2、struts2源码分析

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

 <filter> 
     <filter-name>struts2filter-name>  
     <filterclass>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
     filter-class> 
 filter>  
 <filter-mapping> 
     <filter-name>struts2filter-name>  
     <url-pattern>/*url-pattern> 
 filter-mapping>

注:在早期的struts2中,都是使用FilterDispathcer,从Struts 2.1.3开始,它已不推荐使用。升级到StrutsPrepareAndExecuteFilter。

StrutsPrepareAndExecuteFilter中的方法:
struts2请求过程源码分析_第3张图片

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

 public void init(FilterConfig filterConfig) throws ServletException {
          InitOperations init = new InitOperations();
          Dispatcher dispatcher = null;
          try {
              //封装filterConfig,其中有个主要方法getInitParameterNames将参数名字以String格式存储在List中
              FilterHostConfig config = new FilterHostConfig(filterConfig);
              //初始化struts内部日志
              init.initLogging(config);
              //创建dispatcher ,并初始化
              dispatcher = init.initDispatcher(config);
              init.initStaticContentLoader(config, dispatcher);
             //初始化类属性:prepare 、execute
              prepare = new PrepareOperations(filterConfig.getServletContext(), dispatcher);
              execute = new ExecuteOperations(filterConfig.getServletContext(), dispatcher);
              this.excludedPatterns = init.buildExcludedPatternsList(dispatcher);
             //回调空的postInit方法
              postInit(dispatcher, filterConfig);
         } finally {
              if (dispatcher != null) {
                 dispatcher.cleanUpAfterInit();
             }
             init.cleanup();
         }
     }

关于封装的参数filterConfig,首先看下FilterHostConfig ,源码如下:

  public class FilterHostConfig implements HostConfig {

      private FilterConfig config;
      //构造方法
      public FilterHostConfig(FilterConfig config) {
          this.config = config;
      }
      //根据init-param配置的param-name获取param-value的值  
      public String getInitParameter(String key) {
         return config.getInitParameter(key);
     }
     //返回初始化参数名的迭代器 
     public Iterator getInitParameterNames() {
         return MakeIterator.convert(config.getInitParameterNames());
     }
     //返回Servlet上下文
     public ServletContext getServletContext() {
         return config.getServletContext();
     }
 }

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

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

public Dispatcher initDispatcher( HostConfig filterConfig ) {
         Dispatcher dispatcher = createDispatcher(filterConfig);
         dispatcher.init();
         return dispatcher;
     }

创建Dispatcher,会读取 filterConfig 中的配置信息,将配置信息解析出来,封装成为一个Map,然后根绝servlet上下文和参数Map构造Dispatcher :

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

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

 public void init() {

         if (configurationManager == null) {
             configurationManager = createConfigurationManager(BeanSelectionProvider.DEFAULT_BEAN_NAME);
         }

         try {
             init_FileManager();
             //加载org/apache/struts2/default.properties
            init_DefaultProperties(); // [1]
            //加载struts-default.xml,struts-plugin.xml,struts.xml
            init_TraditionalXmlConfigurations(); // [2]
            init_LegacyStrutsProperties(); // [3]
            //用户自己实现的ConfigurationProviders类 
            init_CustomConfigurationProviders(); // [5]
            //Filter的初始化参数 
            init_FilterInitParameters() ; // [6]
            init_AliasStandardObjects() ; // [7]

            Container container = init_PreloadConfiguration();
            container.inject(this);
            init_CheckWebLogicWorkaround(container);

            if (!dispatcherListeners.isEmpty()) {
                for (DispatcherListener l : dispatcherListeners) {
                    l.dispatcherInitialized(this);
                }
            }
        } catch (Exception ex) {
            if (LOG.isErrorEnabled())
                LOG.error("Dispatcher initialization failed", ex);
            throw new StrutsException(ex);
        }
    }

这里主要是加载一些配置文件的,将按照顺序逐一加载: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方法,该方法内容如下:

public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {

          HttpServletRequest request = (HttpServletRequest) req;
          HttpServletResponse response = (HttpServletResponse) res;

          try {
              //设置编码和国际化
              prepare.setEncodingAndLocale(request, response);
              //创建action上下文
             prepare.createActionContext(request, response);
             prepare.assignDispatcherToThread();
             if (excludedPatterns != null && prepare.isUrlExcluded(request, excludedPatterns)) {
                 chain.doFilter(request, response);
             } else {
                 request = prepare.wrapRequest(request);
                 ActionMapping mapping = prepare.findActionMapping(request, response, true);
                 //如果mapping为空,则认为不是调用action,会调用下一个过滤器链,直到获取到mapping才调用action
                 if (mapping == null) {
                     boolean handled = execute.executeStaticResourceRequest(request, response);
                     if (!handled) {
                         chain.doFilter(request, response);
                     }
                 } else {
                     //执行action
                     execute.executeAction(request, response, mapping);
                 }
             }
         } finally {
             prepare.cleanupRequest(request);
         }
     }

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

  1. prepare.setEncodingAndLocale(request, response)
    第8行是调用prepare对象的setEncodingAndLocale方法,prepare是PrepareOperations类的对象,PrepareOperations类是用来做请求准备工作的。我们看下PrepareOperations类中的setEncodingAndLocale方法:
public void setEncodingAndLocale(HttpServletRequest request, HttpServletResponse response) {
         dispatcher.prepare(request, response);
     }

在这方法里面我们可以看到它只是调用了dispatcher的prepare方法而已,下面我们看看dispatcher的prepare方法:

public void prepare(HttpServletRequest request, HttpServletResponse response) {
          String encoding = null;
          if (defaultEncoding != null) {
              encoding = defaultEncoding;
          }
          // check for Ajax request to use UTF-8 encoding strictly http://www.w3.org/TR/XMLHttpRequest/#the-send-method
          if ("XMLHttpRequest".equals(request.getHeader("X-Requested-With"))) {
              encoding = "UTF-8";
          }

         Locale locale = null;
         if (defaultLocale != null) {
             locale = LocalizedTextUtil.localeFromString(defaultLocale, request.getLocale());
         }

         if (encoding != null) {
             applyEncoding(request, encoding);
         }

         if (locale != null) {
             response.setLocale(locale);
         }

         if (paramsWorkaroundEnabled) {
             request.getParameter("foo"); // simply read any parameter (existing or not) to "prime" the request
         }
     }

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

  1. 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类中时如下定义的:

 static ThreadLocal actionContext = new ThreadLocal();

我们看下PrepareOperations类的createActionContext方法:

 /**
       * Creates the action context and initializes the thread local
       */
      public ActionContext createActionContext(HttpServletRequest request, HttpServletResponse response) {
          ActionContext ctx;
          Integer counter = 1;
          Integer oldCounter = (Integer) request.getAttribute(CLEANUP_RECURSION_COUNTER);
          if (oldCounter != null) {
              counter = oldCounter + 1;
         }
         //此处是从ThreadLocal中获取此ActionContext变量
         ActionContext oldContext = ActionContext.getContext();
         if (oldContext != null) {
             // detected existing context, so we are probably in a forward
             ctx = new ActionContext(new HashMap(oldContext.getContextMap()));
         } else {
             ValueStack stack = dispatcher.getContainer().getInstance(ValueStackFactory.class).createValueStack();
             stack.getContext().putAll(dispatcher.createContextMap(request, response, null, servletContext));
             //stack.getContext()返回的是一个Map,根据此Map构造一个ActionContext
             ctx = new ActionContext(stack.getContext());
         }
         request.setAttribute(CLEANUP_RECURSION_COUNTER, counter);
         //将ActionContext存到ThreadLocal 
         ActionContext.setContext(ctx);
         return ctx;
     }

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

  public Map<String,Object> createContextMap(HttpServletRequest request, HttpServletResponse response,
              ActionMapping mapping, ServletContext context) {

          // request map wrapping the http request objects
          Map requestMap = new RequestMap(request);

          // parameters map wrapping the http parameters.  ActionMapping parameters are now handled and applied separately
          Map params = new HashMap(request.getParameterMap());

         // session map wrapping the http session
         Map session = new SessionMap(request);

         // application map wrapping the ServletContext
         Map application = new ApplicationMap(context);
         //requestMap、params、session等Map封装成为一个上下文Map 
         Map<String,Object> extraContext = createContextMap(requestMap, params, session, application, request, response, context);

         if (mapping != null) {
             extraContext.put(ServletActionContext.ACTION_MAPPING, mapping);
         }
         return extraContext;
     }
  1. request = prepare.wrapRequest(request)
    我们再次回到StrutsPrepareAndExecuteFilter的doFilter方法中,看到第15行:request = prepare.wrapRequest(request);这一句是对request进行包装的,我们看下prepare的wrapRequest方法:
      
public HttpServletRequest wrapRequest(HttpServletRequest oldRequest) throws ServletException {
          HttpServletRequest request = oldRequest;
          try {
              // Wrap request first, just in case it is multipart/form-data
              // parameters might not be accessible through before encoding (ww-1278)
              request = dispatcher.wrapRequest(request, servletContext);
          } catch (IOException e) {
              throw new ServletException("Could not wrap servlet request with MultipartRequestWrapper!", e);
          }
         return request;
     }

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

public HttpServletRequest wrapRequest(HttpServletRequest request, ServletContext servletContext) throws IOException {
          // don't wrap more than once
          if (request instanceof StrutsRequestWrapper) {
              return request;
          }

          String content_type = request.getContentType();
          //如果content_type是multipart/form-data类型,则将request包装成MultiPartRequestWrapper对象,否则包装成StrutsRequestWrapper对象
          if (content_type != null && content_type.contains("multipart/form-data")) {
             MultiPartRequest mpr = getMultiPartRequest();
             LocaleProvider provider = getContainer().getInstance(LocaleProvider.class);
             request = new MultiPartRequestWrapper(mpr, request, getSaveDir(servletContext), provider);
         } else {
             request = new StrutsRequestWrapper(request, disableRequestAttributeValueStackLookup);
         }

         return request;
     }

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

  1. 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方法:

 public ActionMapping findActionMapping(HttpServletRequest request, HttpServletResponse response, boolean forceLookup) {
          //首先从request对象中取mapping对象,看是否存在
          ActionMapping mapping = (ActionMapping) request.getAttribute(STRUTS_ACTION_MAPPING_KEY);
          //不存在就创建一个
          if (mapping == null || forceLookup) {
              try {
                  //首先创建ActionMapper对象,通过ActionMapper对象创建mapping对象
                  mapping = dispatcher.getContainer().getInstance(ActionMapper.class).getMapping(request, dispatcher.getConfigurationManager());
                  if (mapping != null) {
                     request.setAttribute(STRUTS_ACTION_MAPPING_KEY, mapping);
                 }
             } catch (Exception ex) {
                 dispatcher.sendError(request, response, servletContext, HttpServletResponse.SC_INTERNAL_SERVER_ERROR, ex);
             }
         }

         return mapping;
     }

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

  public ActionMapping getMapping(HttpServletRequest request, ConfigurationManager configManager) {
          ActionMapping mapping = new ActionMapping();
          //获得请求的uri,即请求路径URL中工程名以后的部分,如/userAction.action
          String uri = getUri(request);
          //修正url的带;jsessionid 时找不到的bug
          int indexOfSemicolon = uri.indexOf(";");
          uri = (indexOfSemicolon > -1) ? uri.substring(0, indexOfSemicolon) : uri;
          //删除扩展名,如.action或者.do
          uri = dropExtension(uri, mapping);
         if (uri == null) {
             return null;
         }
         //从uri中分离得到请求的action名、命名空间。
         parseNameAndNamespace(uri, mapping, configManager);
         //处理特殊的请求参数,将这些参数存储在一个HashSet中
         handleSpecialParameters(request, mapping);
         //如果允许动态方法调用,即形如/userAction!getAll.action的请求,分离action名和方法名
         return parseActionName(mapping);
     }

可以看到,该方法从请求的uri中分离出name和namespace并存放在mapping中。并对请求action名解析,分离出action和方法名。
5. execute.executeAction(request, response, mapping)
上面我们分析完了mapping的获取,继续看doFilter方法:

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

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

 public void executeAction(HttpServletRequest request, HttpServletResponse response, ActionMapping mapping) throws ServletException {
         dispatcher.serviceAction(request, response, servletContext, mapping);
     }

我们可以看到它里面只是简单的调用了dispatcher的serviceAction方法:我们找到dispatcher的serviceAction方法:

public void serviceAction(HttpServletRequest request, HttpServletResponse response, ServletContext context,
                              ActionMapping mapping) throws ServletException {
        //封转上下文环境,主要将requestMap、params、session等Map封装成为一个上下文Map
        Map<String, Object> extraContext = createContextMap(request, response, mapping, context);

        // If there was a previous value stack, then create a new copy and pass it in to be used by the new Action
        ValueStack stack = (ValueStack) request.getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
        boolean nullStack = stack == null;
        if (nullStack) {
            ActionContext ctx = ActionContext.getContext();
            if (ctx != null) {
                stack = ctx.getValueStack();
            }
        }
        if (stack != null) {
            extraContext.put(ActionContext.VALUE_STACK, valueStackFactory.createValueStack(stack));
        }

        String timerKey = "Handling request from Dispatcher";
        try {
            UtilTimerStack.push(timerKey);
            String namespace = mapping.getNamespace();//从mapping对象获取命名空间
            String name = mapping.getName();          //获取请求的action名
            String method = mapping.getMethod();      //获取请求方法
            //得到配置对象
            Configuration config = configurationManager.getConfiguration();
            //根据执行上下文参数,命名空间,名称等创建用户自定义Action的代理对象  
            ActionProxy proxy = config.getContainer().getInstance(ActionProxyFactory.class).createActionProxy(
                    namespace, name, method, extraContext, true, false);

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

            // if the ActionMapping says to go straight to a result, do it!
            //如果配置文件中执行的这个action配置了result,就直接转到result
            if (mapping.getResult() != null) {
                Result result = mapping.getResult();
                result.execute(proxy.getInvocation());
            } else {
                proxy.execute();
            }

            // If there was a previous value stack then set it back onto the request
            if (!nullStack) {
                request.setAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY, stack);
            }
        } catch (ConfigurationException e) {
            // WW-2874 Only log error if in devMode
            if (devMode) {
                String reqStr = request.getRequestURI();
                if (request.getQueryString() != null) {
                    reqStr = reqStr + "?" + request.getQueryString();
                }
                LOG.error("Could not find action or result\n" + reqStr, e);
            } else {
                if (LOG.isWarnEnabled()) {
                    LOG.warn("Could not find action or result", e);
                }
            }
            sendError(request, response, context, HttpServletResponse.SC_NOT_FOUND, e);
        } catch (Exception e) {
            if (handleException || devMode) {
                sendError(request, response, context, HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e);
            } else {
                throw new ServletException(e);
            }
        } finally {
            UtilTimerStack.pop(timerKey);
        }
    }

核心代码就是根据前面的请求准备工作,为用户创建ActionProxy对象,执行相应的action,若配置有执行结果,那么返回执行结果。

你可能感兴趣的:(项目开发)