最详细的SpringMVC执行流程

SpringMVC执行流程图

1.png

在前端控制器中最最重要的方法是 doDispatch,在这个方法中 ,起到委派模式中委派者的角色,负责把 任务分发给各个角色做处理

分发的主要任务:

  1. 获取处理器映射器

  2. 根据处理器映射器获取处理器适配器

  3. 根据处理器适配器获取视图ModelAndView

  4. 使用视图解析解解析视图

  5. 渲染视图

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
        //把request对象赋值给 processedRequest
        HttpServletRequest processedRequest = request;
        //定义 HandlerExecutionChain 如果是访问controller方法的话,封装方法对象(方法对象中封装了controller对象),  HandlerExecutionChain还将封装所有的拦截器
        HandlerExecutionChain mappedHandler = null;
        boolean multipartRequestParsed = false;

        WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);

        try {
            //将会通过 ha.handle(processedRequest, response, mappedHandler.getHandler()); 获取 ModelAndView
            ModelAndView mv = null;
            Exception dispatchException = null;

            try {
                // 1.尝试将当前请求转换为MultipartHttpServletRequest
                processedRequest = checkMultipart(request);
                multipartRequestParsed = (processedRequest != request);

                
        // 2.查找当前请求对应的handler,包括Handler(控制器也就是controller本身)本身和Handler拦截器

        //当遍历到requestMappingHandlerMapping时  在requestMappingHandlerMapping中其实存储了  所有拦截器的对象
                /**
                 * 这个方法一路千辛万苦,一路封装,满载而归
                 *  1.首先是调用AbstractHandlerMapping的getHandler方法,然后调用  AbstractHandlerMethodMapping的getHandlerInternal方法
                 *  然后重新创建创建对象 new HandlerMethod(this, handler) 把controller对象(从工厂中获取) 赋值 给方法对象HandlerMethod
                 *  所以第一步就是: 把controller对象赋值给方法对象
                 *
                 *  2.然后调用AbstractHandlerMapping.getHandlerExecutionChain 转换为 HandlerExecutionChain 对象
                 *  遍历拦截器集合 把所有的拦截器对象赋值给HandlerExecutionChain对象
                 *  所以第二步就是: 把方法对象转换为HandlerExecutionChain对象并把所有的拦截器赋值到其中
                 *
                 */
                //根据请求request对象,调用处理器映射器寻找处理器,其实就是 HandlerExecutionChain 对象
                mappedHandler = getHandler(processedRequest);
                /**
                 * 此时的 mappedHandler 即为  HandlerExecutionChain 对象
                 * HandlerExecutionChain 对象中 封装了浏览器访问的方法对应的方法对象,方法对象中封装了对象的controller对象,HandlerExecutionChain封装了所有的拦截器
                 */


                // 未能找到对应的handler,抛出NoHandlerFoundException异常并返回404
                if (mappedHandler == null) {
                    noHandlerFound(processedRequest, response);
                    return;
                }

                // Determine handler adapter for the current request.
                // 3.查找当前请求对应的HandlerAdapter
                //把方法对象传进去,获取到一个适配器
                HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

                // Process last-modified header, if supported by the handler.
                // 4.处理last-modified请求头,如果当前请求支持的话
                //获取方法的请求方法
                String method = request.getMethod();

                boolean isGet = "GET".equals(method);
                if (isGet || "HEAD".equals(method)) {
                    long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
                    if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
                        return;
                    }
                }

                // 5.应用前置拦截器
                // 如果有拦截器返回false,则表明该拦截器已经处理了返回结果,直接返回;  
                //注意:   此时的 processedRequest 其实就是request 对象
                //就是在判断定义所有的拦截器 的前置方法,返回的到底是true,还是false
                //如果有一个前置返回的是false,那么停止执行下面的代码,  只有所有的拦截器的前置方法返回的true才可以
                if (!mappedHandler.applyPreHandle(processedRequest, response)) {
                    return;
                }


                // Actually invoke the handler.
                // 6.调用HandlerAdapter的handler方法,真正开始处理Controller

                //在以上的所有步骤中,   ModelAndView都还没有返回
                //这个方法尝试获取  ModelAndView 对象   把request对象 ,response对象 和方法对象传进去

                //进入到RequestMappingHandlerAdaper适配器的handleInternal

                //准备获取ModelAndView对象 同时在方法里面执行了controller方法的内容
                mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
                //此时ModelAndView对象的view值为跳转的路径

                // 7.如果当前请求是并发处理,直接返回
                if (asyncManager.isConcurrentHandlingStarted()) {
                    return;
                }

                // 8. 如果当前返回值中不包含视图名的话,为返回值设定默认视图名,
                //意思如果你没有设置跳转路径的话,这个方法默认给你加跳转路径
                applyDefaultViewName(processedRequest, mv);

                // 9.应用已注册拦截器的后置方法。
                //倒着遍历所有的拦截器 先注册的后执行
            mappedHandler.applyPostHandle(processedRequest, response, mv);
        }
            catch (Exception ex) {
                dispatchException = ex;
            }
            catch (Throwable err) {
                // As of 4.3, we're processing Errors thrown from handler methods as well,
                // making them available for @ExceptionHandler methods and other scenarios.
                dispatchException = new NestedServletException("Handler dispatch failed", err);
            }
            // 10.处理分发调用结果,如视图模型解析、返回等工作   如果以上有任何错误,把错误信息封装赋值给dispatchException 错误对象
            //如果dispatchException不为空的话,打印错误信息,如果ModelAndView返回的是一个页面的话,会重新发起请求
            //如果没有ModelAndView为空了 ,或者说controller返回不是一个页面了, 执行拦截器的后置方法,也是倒着遍历
            //在这里面还干了一个一件事 ,那就是获取到了 view视图对象
            processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
        }
        catch (Exception ex) {
            triggerAfterCompletion(processedRequest, response, mappedHandler, ex);
        }
        catch (Throwable err) {
            triggerAfterCompletion(processedRequest, response, mappedHandler,
                    new NestedServletException("Handler processing failed", err));
        }
        finally {
            if (asyncManager.isConcurrentHandlingStarted()) {
                // Instead of postHandle and afterCompletion
                if (mappedHandler != null) {
                    mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
                }
            }
            else {
                //整理由多部分请求使用的任何资源。
                // Clean up any resources used by a multipart request.
                if (multipartRequestParsed) {
                    cleanupMultipart(processedRequest);
                }
            }
        }
    }

聊一下比较重要的几个方法

1.获取处理器映射器

mappedHandler = getHandler(processedRequest);

在doDispatch方法中找这个方法,按住Ctrl键点击进入

此一路千辛万苦,一路封装,满载而归

@Nullable
    protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
        //handlerMappings中有5个对象
        //1. favconHandlerMapping 2.requestMappingHandlerMapping 3.beanNameHandlerMapping 4.resourceHandlerMapping 5.welcomePageHandlerMapping
        //遍历这个五个对象
        if (this.handlerMappings != null) {
            for (HandlerMapping mapping : this.handlerMappings) {
                //当遍历到requestMappingHandlerMapping时  在requestMappingHandlerMapping中其实存储了  所有拦截器的对象

                /**
                 * 这个方法一路千辛万苦,一路封装,满载而归
                 *  1.首先是调用AbstractHandlerMapping的getHandler方法,然后调用  AbstractHandlerMethodMapping的getHandlerInternal方法
                 *  然后重新创建创建对象 new HandlerMethod(this, handler) 把controller对象(从工厂中获取) 赋值 给方法对象HandlerMethod
                 *  所以第一步就是: 把controller对象赋值给方法对象
                 *
                 *  2.然后调用AbstractHandlerMapping.getHandlerExecutionChain 转换为 HandlerExecutionChain 对象
                 *  遍历拦截器集合 把拦截器所有的对象赋值给HandlerExecutionChain对象的集合
                 *   所以第二步就是: 把方法对象转换为HandlerExecutionChain对象并把所有的拦截器赋值到其中
                 *
                 */
                HandlerExecutionChain handler = mapping.getHandler(request);
                if (handler != null) {
                    return handler;
                }
            }
        }
        return null;
    }

我们再看一下

mapping.getHandler(request);方法

按住Ctrl键点击进入,发现是HandlerMapping接口 , 按住快捷键 Ctrl + Alt + B 选择AbstractHandlerMapping 这个实现类

@Override
    @Nullable
    public final HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
        //获取Controller对象,但是获取到时对应方法的对象,方法对象中封装有controller对象
        //一路获取controller对象,把controller对象封装进方法对象中,(前提是访问的是controller中的方法)
        Object handler = getHandlerInternal(request);
        if (handler == null) {
            handler = getDefaultHandler();
        }
        if (handler == null) {
            return null;
        }
        // Bean name or resolved handler?
        if (handler instanceof String) {
            String handlerName = (String) handler;
            handler = obtainApplicationContext().getBean(handlerName);
        }

         //把方法对象和request传进去 ,准备把所有的拦截器封装进 HandlerExecutionChain对象中
        HandlerExecutionChain executionChain = getHandlerExecutionChain(handler, request);

      // 现在 HandlerExecutionChain 对象中有 方法对象 而方法对象中存储了controller对象, HandlerExecutionChain中有所有的拦截器对象
        if (logger.isTraceEnabled()) {
            logger.trace("Mapped to " + handler);
        }
        else if (logger.isDebugEnabled() && !request.getDispatcherType().equals(DispatcherType.ASYNC)) {
            logger.debug("Mapped to " + executionChain.getHandler());
        }

        if (CorsUtils.isCorsRequest(request)) {
            CorsConfiguration globalConfig = this.corsConfigurationSource.getCorsConfiguration(request);
            CorsConfiguration handlerConfig = getCorsConfiguration(handler, request);
            CorsConfiguration config = (globalConfig != null ? globalConfig.combine(handlerConfig) : handlerConfig);
            executionChain = getCorsHandlerExecutionChain(request, executionChain, config);
        }

        return executionChain;
    }

2.获取处理器适配器

HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

我们再回到 doDispatch方法中,找到该方法, 按住Ctrl键点击进入

    //参数为对应方法对象,或者是页面资源对象
    protected HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
        //handlerAdapters 有三个适配器对象
        //1.RquestMappingHandlerAdapter  2.HttpRequestHandlerAdapter 3.SimpleControllerHandlerAdapter
        if (this.handlerAdapters != null) {
            for (HandlerAdapter adapter : this.handlerAdapters) {
               //查看哪个处理器符合请求需要的,返回符合条件的处理器
                if (adapter.supports(handler)) {
                    return adapter;
                }
            }
        }
        throw new ServletException("No adapter for handler [" + handler +
                "]: The DispatcherServlet configuration needs to include a HandlerAdapter that supports this handler");
    }

3.获取视图ModelAndView

mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

我们再回到 doDispatch方法中,找到该方法, 点击进入

发现是一个HandlerAdapter接口 ,按住快捷键 Ctrl + Alt + B ,选择 AbstractHandlerMethodAdaper

按住Ctrl键点击 handleInternal 方法,发现是一个抽象的方法,再次 按住快捷键 Ctrl + Alt + B 进入到RequestMappingHandlerAdapter

@Override
    protected ModelAndView handleInternal(HttpServletRequest request,
            HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {

        ModelAndView mav;
        checkRequest(request);

        //如果需要,在同步块中执行InvokehandlerMethod。
        // Execute invokeHandlerMethod in synchronized block if required.
        if (this.synchronizeOnSession) {
            HttpSession session = request.getSession(false);
            if (session != null) {
                Object mutex = WebUtils.getSessionMutex(session);
                synchronized (mutex) {
                    mav = invokeHandlerMethod(request, response, handlerMethod);
                }
            }
            else {
                // No HttpSession available -> no mutex necessary
                mav = invokeHandlerMethod(request, response, handlerMethod);
            }
        }
        else {
            // No synchronization on session demanded at all...
            //尝试获取ModelAndView对象,如果没有HTML,则获取不到视图
            //这个方法里面执行了controller的内容
            mav = invokeHandlerMethod(request, response, handlerMethod);
        }

        if (!response.containsHeader(HEADER_CACHE_CONTROL)) {
            if (getSessionAttributesHandler(handlerMethod).hasSessionAttributes()) {
                applyCacheSeconds(response, this.cacheSecondsForSessionAttributeHandlers);
            }
            else {
                prepareResponse(response);
            }
        }
          //获取到ModelAndView对象
        return mav;
    }

4.解析视图

processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);

我们再回到 doDispatch方法中,找到该方法, 按住Ctrl键点击进入

private void processDispatchResult(HttpServletRequest request, HttpServletResponse response,
            @Nullable HandlerExecutionChain mappedHandler, @Nullable ModelAndView mv,
            @Nullable Exception exception) throws Exception {

        boolean errorView = false;

        if (exception != null) {
            if (exception instanceof ModelAndViewDefiningException) {
                logger.debug("ModelAndViewDefiningException encountered", exception);
                mv = ((ModelAndViewDefiningException) exception).getModelAndView();
            }
            else {
                Object handler = (mappedHandler != null ? mappedHandler.getHandler() : null);
                mv = processHandlerException(request, response, handler, exception);
                errorView = (mv != null);
            }
        }

        //处理程序是否返回要呈现的视图?
        // Did the handler return a view to render?
        if (mv != null && !mv.wasCleared()) {
            //这个方法当中会获取到视图view
            render(mv, request, response);
            if (errorView) {
                WebUtils.clearErrorRequestAttributes(request);
            }
        }
        else {
            if (logger.isTraceEnabled()) {
                logger.trace("No view rendering, null ModelAndView returned.");
            }
        }

        if (WebAsyncUtils.getAsyncManager(request).isConcurrentHandlingStarted()) {
            // Concurrent handling started during a forward
            return;
        }

        if (mappedHandler != null) {
            //如果没有ModelAndView为空了 ,或者说controller返回不是一个页面了,执行拦截器的后置方法,也是倒着遍历
            mappedHandler.triggerAfterCompletion(request, response, null);
        }
    }

找到 render(mv, request, response); 方法, 按住Ctrl键点击进入

protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception {
        // Determine locale for request and apply it to the response.
        Locale locale =
                (this.localeResolver != null ? this.localeResolver.resolveLocale(request) : request.getLocale());
        response.setLocale(locale);

        View view;
        String viewName = mv.getViewName();
        if (viewName != null) {
            //我们需要解析视图名称
            // We need to resolve the view name.
            //获取到了视图
            view = resolveViewName(viewName, mv.getModelInternal(), locale, request);
            if (view == null) {
                throw new ServletException("Could not resolve view with name '" + mv.getViewName() +
                        "' in servlet with name '" + getServletName() + "'");
            }
        }
        else {
            // No need to lookup: the ModelAndView object contains the actual View object.
            view = mv.getView();
            if (view == null) {
                throw new ServletException("ModelAndView [" + mv + "] neither contains a view name nor a " +
                        "View object in servlet with name '" + getServletName() + "'");
            }
        }

        // Delegate to the View object for rendering.
        if (logger.isTraceEnabled()) {
            logger.trace("Rendering view [" + view + "] ");
        }
        try {
            if (mv.getStatus() != null) {
                response.setStatus(mv.getStatus().value());
            }
            //获取到视图之后
            view.render(mv.getModelInternal(), request, response);
        }
        catch (Exception ex) {
            if (logger.isDebugEnabled()) {
                logger.debug("Error rendering view [" + view + "]", ex);
            }
            throw ex;
        }
    }

这个方法中,解析视图ModelAndView ,获取View对象

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容