SprigMVC源码解析

 
1.SpringMVC入口在哪里?
    入口在web.xml中
 
2.加载DispatcherServlet
    1.首先我们看一下继承树
    2.很明显SpringMVC也是通过继承HttpServlet来实现请求的接收的
        2.1 那么他继承了HttpServlet那么肯定会有doGet();doPost什么的对吧?
        2.2 那么doGet();doPost()在哪里被实现的呢?我们可以看一下他的子类,通过查找他是被抽象类FrameworkServlet 所实现的
            
        2.3 通过查看FrameworkServlet 中的doGet或者doPost发现都调用了processRequest(request,response)方法
        2.4 通过查看processRequest(request,response)方法还是在FrameworkServlet 所实现
            
            这个方法重点我们看try里面的方法doService
        2.5 doService在哪儿呢?这个方法被FormeworkServlet抽象类 做成了接口,被DispatcherServlet实现了
            
            
            重点看到标记的方法  这个方法是重点!!!重点!!!重点!!!
        2.6 doDispatch(request,response)这个方法在哪儿呢?这个方法还是在DispatcherServlet中
        代码解析:
        
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
        HttpServletRequest processedRequest = request;
        HandlerExecutionChain mappedHandler = null;
        boolean multipartRequestParsed = false;
 
 
        // 异步管理器
        WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
 
 
        try {
            // 定义ModelAndView对象
            ModelAndView mv = null;
            // 定义异常对象
            Exception dispatchException = null;
 
 
            try {
                // 检查是否是文件上传请求
                processedRequest = checkMultipart(request);
                multipartRequestParsed = (processedRequest != request);
 
 
                // 获取当前请求的处理器  ==> 也就是我们自己定义的handler 等分析这个方法的时候在分析
                mappedHandler = getHandler(processedRequest);
                // 如果没有找到则调用一个404没有找到则 这个方法主要就是如果你自己定义了的404 某则抛出404的异常
                if (mappedHandler == null || mappedHandler.getHandler() == null) {
                    noHandlerFound(processedRequest, response);
                    return;
                }
 
 
                // 获取处理当前请求的适配器(反射工具) 比如说是xml配置的还是注解注释的 等分析这个方法的时候在分析
                HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
 
 
                // 如果是get就做一些缓存
                String method = request.getMethod();
                boolean isGet = "GET".equals(method);
                if (isGet || "HEAD".equals(method)) {
                    long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
                    if (logger.isDebugEnabled()) {
                        logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " + lastModified);
                    }
                    if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
                        return;
                    }
                }
 
 
                if (!mappedHandler.applyPreHandle(processedRequest, response)) {
                    return;
                }
 
 
                // 通过适配器也就是反射工具 真正的执行我们编写的方法 并且返回ModelAndView
                mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
                
                // 是否是异步请求 如果是直接返回
                if (asyncManager.isConcurrentHandlingStarted()) {
                    return;
                }
 
 
                // 如果目标方法没有返回值 就使用默认的视图名
                applyDefaultViewName(processedRequest, mv);
                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);
            }
            // 返回页面的方法 根据ModelAndView转发到对应的页面 而且ModelAndView中的数据可以从请求域中取到
            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);
                }
            }
        }
    }
        2.7 重要的几个方法解析
            在分析这个方法要掌握9大组件!!! 虽然我也没掌握 但是正在学习
            组件的位置在DispatcherServlet中的属性
            
            
              getHandler ( processedRequest )
 getHandlerAdapter ( mappedHandler.getHandler ())
mv = ha.handle ( processedRequest , response , mappedHandler.getHandler ()) ;
processDispatchResult ( processedRequest , response , mappedHandler , mv , dispatchException ) ;
 
            1.getHandler
                
                我们可以看到这个中都是我们是用@Requestmapping编写的请求路径  他会进行比对之后返回那个handler来执行
                那么这个实在哪里初始化的呢?是Spring做的init,这就是初始化的9大组件的地方!!!
                
                
            2.getHanleAdapter
            
            这个第一个点是用Spring初始化的组件  他会通过handler的类型看是否符合 最后返回适配器
            3.handler
            
                    这里大量是用了反射 来进行操作!
            4.processDispatchResult
            
            首先看看是否有异常,然后准备一些数据什么的,然后返回页面
如有不对请评论指出 , 我也是刚刚了解框架底层
 
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值