tomcat与springmvc 结合 之---第16篇 servlet如何解析成员变量和DispatcherServlet如何解析

writedby 张艳涛,用了两个星期将深入刨析tomcat看完了,那么接下来该看什么呢?真是不知道,知识这东西上一个月看的jvm,锁.多线程并发 又都忘了....

tomcat学完,我打算看springmvc因为,spring本质就是一个servlet, 叫DispatcherServlet,那么俩者联系紧密,打算结合二者,进行学习

昨天看了一天发现spring源码,看起来比tomcat要难,因为springmv太杂了

以前看过知道web.xml中的<servlet>标签解析,那么遇到了

 <servlet>
        <servlet-name>loginServlet</servlet-name>
        <servlet-class>com.qcc.study.servlet02.LoginServlet</servlet-class>
        <!-- 配置Servlet初始化参数 -->
        <init-param>
            <param-name>initParam</param-name>
            <param-value>qcc</param-value>
        </init-param>
        <!-- Web容器启动时就加载并实例化该Servlet -->
        <load-on-startup>0</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>loginServlet</servlet-name>
        <url-pattern>/login</url-pattern>    
    </servlet-mapping>
</servlet>

以上的参数该如何解析呢?这个init-param目的就是要给成员变量初始化值,看如何使用

package com.qcc.study.servlet02;
 
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
 
public class LoginServlet extends HttpServlet {
     成员变量1 initParam = null;
    成员变量2=null;
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //获取servlet初始化参数: String initParam = getServletConfig().getInitParameter("initParam"); System.out.println("initParam: ---->" + initParam); }

那么可以看到,

<init-parm>配置在<servlet>标签中,用来初始化当前的Servlet的,属于当前Servlet的配置,因此存放在 servletConfig对象中;
通过getServletConfig().getInitParameter("initParam")的方式获取;
如果通过源码来看webruleset中对"web-app/servlet/init-param" 解析

看 addInitParameter StandardWrapper中

 

可以看到其实参数以hashmap的方法来组织成键值对,如果使用的时候,实际上是从wrapper容器中取得的,和servlet.class没关系

如果在形成的servlet对象的成员变量赋值写在init()方法里面,最后形成的servlet对象的成员变量就是完全体了,这其实就是dispatchServlet的做法


先看一个dispatchservlet的配置文件web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xmlns="http://java.sun.com/xml/ns/javaee"
         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
         id="WebApp_ID" version="3.0">
    <display-name>springmvcFirst</display-name>
    <!-- springMVC前端控制器 -->
    <servlet>
        <servlet-name>springmvc</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <!-- contextConfigLocation配置springmvc加载的配置文件(配置处理器、映射器等) 如果不配置contextConfigLocation,默认加载的是/WEB-INF/servlet名称-servlet.xml(springmvc-servlet.xml) -->
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springmvc.xml</param-value>
        </init-param>
    </servlet>
    <servlet-mapping>
        <servlet-name>springmvc</servlet-name>
        <!-- url-pattern:*.action的请交给DispatcherServlet处理。 -->
        <url-pattern>*.action</url-pattern>
    </servlet-mapping>

</web-app>

看这其实是给servlet设值,那么看dispatchservlet的成员变量

重点看FrameworkServlet,是dispatcherServlet的父类, 其中有一个成员变量叫contextConfigLocation,对应了标签中的参数名

那么在哪里赋值的呢?

在HttpServletBean的init()方法里面

其中init(servletConfig)的调用逻辑 

GenericServlet===>

    public void init(ServletConfig config) throws ServletException {
    this.config = config;
    this.init();
    }

本类的this.init(),其实这里使用了模板方法,这个this是指的dispatchservlet对象,

    public void init() throws ServletException {

    }

所以调用的init()方法在 HttpServletBean 中

    public final void init() throws ServletException {
        if (logger.isDebugEnabled()) {
            logger.debug("Initializing servlet '" + getServletName() + "'");
        }

        // Set bean properties from init parameters.
        try {
            PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
            BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
            ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
            bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
            initBeanWrapper(bw);
            bw.setPropertyValues(pvs, true);
        }
        catch (BeansException ex) {
            logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
            throw ex;
        }

        // Let subclasses do whatever initialization they like.
        initServletBean();

        if (logger.isDebugEnabled()) {
            logger.debug("Servlet '" + getServletName() + "' configured successfully");
        }
    }

这个方法 在servlet调用getServletConfig()实际上是StandardWrapperFacade,他是StandardWrapper的门面类...

最后能看到BW是通过反射来实现属性的注入的

 


另外说一点,这servlet类的设计,使用了模板方法,是在GenericServlet中的

    public void init(ServletConfig config) throws ServletException {
    this.config = config;
    this.init();
    }

那么看tomcat的调用

 

这个servlet对象就是dispatchservelet,这样子类调用父类的GenericServlet的init(servletconfig)方法,接着调用this(这个this是子类disptacherservlet对象,可以sout(this)验证).init()方法,

整个dispatcherservlet的调用思路就弄清楚了

 

posted @ 2021-07-20 11:27  张艳涛&java  阅读(89)  评论(0编辑  收藏  举报