• how tomcat works 读书笔记 十一 StandWrapper 下


    StandardWrapperValve

    StandardWrapperValve是StandardWrapper的基础阀,主要完毕了三个工作。
    1 调用StandardWrapper的allocate的方法来获得该StandardWrapper所表示的servlet实例
    2 运行与该servelt相关联的所有过滤器
    3 调用servlet的service方法

    当中第二三步能够细分为
      调用它的 private createFilterChain 方法获得过滤链
      调用过滤器链的 doFilter 方法。

    这里面就调用了servlet 的 service方法
      释放过滤器链
      调用包装器的deallocate方法

      假设Servlet无法使用了,调用包装器的 unload 方法


    我们看看大致的代码片段
    // Allocate a servlet instance to process this request
    try {
        if (!unavailable) {
        servlet = wrapper.allocate();
        }
    }
    ...
    // Acknowlege the request
    try {
        response.sendAcknowledgement();
    }
    ...
    // Create the filter chain for this request
        ApplicationFilterChain filterChain = createFilterChain(request,servlet);
    // Call the filter chain for this request// This also calls the servlet's servicet() method
    try {
        String jspFile = wrapper.getJspFile();
        if (jspFile != null)
        sreq.setAttribute(Globals.JSP_FILE_ATTR, jspFile);
        else
        sreq.removeAttribute(Globals.JSP_FILE_ATTR);
        if ((servlet != null) && (filterChain != null)) {
        filterChain.doFilter(sreq, sres);
        }
        sreq.removeAttribute(Globals.JSP_FILE_ATTR);
    }
    ...
    // Release the filter chain (if any) for this request
    try {
        if (filterChain != null)
            filterChain.release();
    }
    ...
    // Deallocate the allocated servlet instance
    try {
        if (servlet != null) {
            wrapper.deallocate(servlet);
        }
    }
    ...
    // If this servlet has been marked permanently unavailable,
    // unload it and release this instance
    try {
        if ((servlet != null) && (wrapper.getAvailable() ==Long.MAX_VALUE)) {
            wrapper.unload();
        }
    }


    看了上面的代码,大家应该能看出来最复杂的部分有两处
    其一 ApplicationFilterChain filterChain = createFilterChain(request,servlet);
    其二 filterChain.doFilter(sreq, sres);
    一步一步来。

    FilterDef类

    这个类的全名应该是FilterDefinition,过滤器描写叙述类。


    里面包括了一个Filter的filterClass,filterName等基本信息,及get/set方法。
    这里面的属性,我们能够看看这个

        /**
         * The set of initialization parameters for this filter, keyed by
         * parameter name.
         */
        private Map<String, String> parameters = new HashMap<String, String>();
    用HashMap存储了初始化參数,它有get方法,添加属性的方法是addInitParameter(String name, String value)。

    ApplicationFilterConfig类

    org.apache.catalina.core.ApplicationFilterConfig 实现了javax.servlet.FilterConfig 接口。ApplicationFilterConfig 负责管理 web应用程序启动的时候创建的过滤器实例。
    其构造函数例如以下:
    public ApplicationFilterConfig(Context context, FilterDef filterDef)
                                     throws ClassCastException, ClassNotFoundException,IllegalAccessException, InstantiationException, ServletException
    在这里我们主要谈谈它的getFilter方法,该方法的功能事实上就是载入过滤器类并初始化它。


    首先从filterDef里面获得filterClass;

            String filterClass = filterDef.getFilterClass();
            ClassLoader classLoader = null;
            if (filterClass.startsWith("org.apache.catalina."))
                classLoader = this.getClass().getClassLoader();
            else
                classLoader = context.getLoader().getClassLoader();
        .....
                Class<?> clazz = classLoader.loadClass(filterClass);
            this.filter = (Filter) clazz.newInstance();
            filter.init(this);
            return (this.filter);
    还是没有什么要说的。


    ApplicationFilterChain类

    StandardWrapperValve 类中的 invoke 方法创建一个该类的实例而且调用它的 doFilter 方法。ApplicationFilterChain类的doFilter(事实上是internalDoFilter)调用该链中第一个过滤器的 doFilter 方法。


    ApplicationFilterChain类中,有一个ApplicationFilterConfig的引用

        private ArrayList<ApplicationFilterConfig> filters = new ArrayList<ApplicationFilterConfig>();

    看到了把,数组形式来存放链条。

    典型的责任链模式。

    Filter 接口中doFilter 方法的签名例如以下:
    public void doFilter(ServletRaquest request, ServletResponse response,FilterChain chain)
                throws java.io.IOException, ServletException
    在ApplicationFilterChain的doFilter方法中,它会将自己作为第三个參数传递给它。
    我们看一个Filter实现类的样例
    public void doFilter(ServletRequest request, ServletResponse response,
                        FilterChain chain) throws IOException, ServletException {
        // do something here
        ...
        chain.doFilter(request, response);
    }
    循环往复了
    此处的循环不是那么easy理解,建议參考鄙人的还有一篇博客

    http://blog.csdn.net/dlf123321/article/details/40078583


    等全部的Filter都运行完了,就是以下的代码

               if ((request instanceof HttpServletRequest) &&
                    (response instanceof HttpServletResponse)) {
                    servlet.service((HttpServletRequest) request,
                                    (HttpServletResponse) response);
                } else {
                    servlet.service(request, response);
                }

    什么时候Filter才算运行完了呢?

     private Iterator<ApplicationFilterConfig> iterator = null;
    
    ....
    
     if (this.iterator == null)
                this.iterator = filters.iterator();
    
            // Call the next filter if there is one
            if (this.iterator.hasNext()) {
    
                        //运行filter
    
           }
    ...
          //调用servlet的service的代码块
    ...
    


    应用程序

    和之前的几章没有什么差别,不再赘述。






  • 相关阅读:
    洛谷P1352没有上司的舞会+树形二维DP
    高精度模板(从洛谷题解中骗来的
    Codeforces#398 &767C. Garland 树形求子节点的和
    LuoGu-P1122 最大子树和+树形dp入门
    HDU-3549Flow Problem 最大流模板题
    Codeforces Round #486 (Div. 3)988E. Divisibility by 25技巧暴力||更暴力的分类
    Codeforces Round #486 (Div. 3)988D. Points and Powers of Two
    数据结构&字符串:01字典树
    数据结构:可持久化平衡树
    数据结构:并查集-拆点
  • 原文地址:https://www.cnblogs.com/lxjshuju/p/7124562.html
Copyright © 2020-2023  润新知