亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

SpringMVC中的DispatcherServlet初始化流程詳解

 更新時(shí)間:2023年12月02日 08:58:14   作者:nuomizhende45  
這篇文章主要介紹了SpringMVC中的DispatcherServlet初始化流程詳解,DispatcherServlet這個(gè)前端控制器是一個(gè)Servlet,所以生命周期和普通的Servlet是差不多的,在一個(gè)Servlet初始化的時(shí)候都會(huì)調(diào)用該Servlet的init()方法,需要的朋友可以參考下

DispatcherServlet初始化流程

分析一下DispatcherServlet這個(gè)前端控制器的啟動(dòng)和初始化的整個(gè)過(guò)程

通過(guò)前面分析已經(jīng)知道了DispatcherServlet這個(gè)前端控制器是一個(gè)Servlet了,所以生命周期和普通的Servlet是差不多的,在一個(gè)Servlet初始化的時(shí)候都會(huì)調(diào)用該Servlet的init()方法。下面這個(gè)是DispatcherSerlvet父類(lèi)HttpServletBean中的init方法。

我們發(fā)現(xiàn)這里會(huì)調(diào)用initServletBean()方法進(jìn)行具體的初始化,而該類(lèi)這個(gè)方法的具體實(shí)現(xiàn)這是其子類(lèi)FrameworkServlet。主要邏輯就是初始化上下文。

    protected final void initServletBean() throws ServletException {
        this.getServletContext().log("Initializing Spring " + this.getClass().getSimpleName() + " \'" + this.getServletName() + "\'");
        if(this.logger.isInfoEnabled()) {
            this.logger.info("Initializing Servlet \'" + this.getServletName() + "\'");
        }
        long startTime = System.currentTimeMillis();
        try {
            /**
            這里初始化上下文
            **/
            this.webApplicationContext = this.initWebApplicationContext();
            this.initFrameworkServlet();
        } catch (RuntimeException | ServletException var4) {
            this.logger.error("Context initialization failed", var4);
            throw var4;
        }
        if(this.logger.isDebugEnabled()) {
            String value = this.enableLoggingRequestDetails?"shown which may lead to unsafe logging of potentially sensitive data":"masked to prevent unsafe logging of potentially sensitive data";
            this.logger.debug("enableLoggingRequestDetails=\'" + this.enableLoggingRequestDetails + "\': request parameters and headers will be " + value);
        }
        if(this.logger.isInfoEnabled()) {
            this.logger.info("Completed initialization in " + (System.currentTimeMillis() - startTime) + " ms");
        }
    }

繼續(xù)跟進(jìn)initWebApplicationContext這個(gè)方法

   protected WebApplicationContext initWebApplicationContext() {
        /**
        若webApplicationContext不為空的時(shí)候從SerlvetContext去出根上下文作為它的雙親上下文
        **/
        WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(this.getServletContext());
        WebApplicationContext wac = null;
        if(this.webApplicationContext != null) {
            wac = this.webApplicationContext;
            if(wac instanceof ConfigurableWebApplicationContext) {
                ConfigurableWebApplicationContext attrName = (ConfigurableWebApplicationContext)wac;
                if(!attrName.isActive()) {
                    if(attrName.getParent() == null) {
                        attrName.setParent(rootContext);
                    }
                    this.configureAndRefreshWebApplicationContext(attrName);
                }
            }
        }
        if(wac == null) {
            wac = this.findWebApplicationContext();
        }
        if(wac == null) {
            wac = this.createWebApplicationContext(rootContext);
        }
        //這里通過(guò)調(diào)用子類(lèi)DispatcherServlet實(shí)現(xiàn)的onRefresh方法
        if(!this.refreshEventReceived) {
            this.onRefresh(wac);
        }
        //把當(dāng)前建立好的上下文存到ServletContext里面去
        if(this.publishContext) {
            String attrName1 = this.getServletContextAttributeName();
            this.getServletContext().setAttribute(attrName1, wac);
        }
        return wac;
    }

繼續(xù)跟進(jìn)這個(gè)onRefresh方法,發(fā)現(xiàn)是一個(gè)模板方法,其具體實(shí)現(xiàn)子類(lèi)則是DispatcherServlet。

跳轉(zhuǎn)到DispatcherServlet中查看該方法,發(fā)現(xiàn)里面又調(diào)用了initStrategies這個(gè)方法

這時(shí)我們就大致了解了,這個(gè)DispatcherServlet初始化的過(guò)程了,首先DispatcherServlet持有者一個(gè)以自己的Servlet名字命名的Ioc容器,也就是我們看到的WebApplicationContext對(duì)象,這個(gè)Ioc容器建立起來(lái)后,與Web容器相關(guān)的各種配置加載也都完成了。并且這個(gè)初始化的入口就是由最初的HttpServletBean的init方法觸發(fā)的,因?yàn)檫@個(gè)HttpServletBean是HttpServlet的子類(lèi),接下來(lái)HttpServletBean的子類(lèi)FrameworkServlet對(duì)Ioc容器進(jìn)行了初始化操作,并且利用onRefresh方法回調(diào)了DispatcherServlet中的initStrategies方法,在這個(gè)方法里啟動(dòng)了整個(gè)SpringMVC框架了,我們繼續(xù)往下面跟進(jìn)看看。

    
//該屬性默認(rèn)為true
    private boolean detectAllHandlerMappings = true;
private void initHandlerMappings(ApplicationContext context) {
        this.handlerMappings = null;
        //這里面的邏輯是從導(dǎo)入所有的HandlerMappingBean,這些Bean有可能存在與雙親容器中,也可能在DispathcerServlet持有的容器的,這里detectAllHandlerMappings默認(rèn)為true,默認(rèn)從所有容器中導(dǎo)入
        if(this.detectAllHandlerMappings) {
            Map hm = BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
            if(!hm.isEmpty()) {
                this.handlerMappings = new ArrayList(hm.values());
                AnnotationAwareOrderComparator.sort(this.handlerMappings);
            }
        } else {
            //否則通過(guò)直接通過(guò)名字從當(dāng)前的IOC容器中通過(guò)getBean方法獲取handlerMapping
            try {
                HandlerMapping hm1 = (HandlerMapping)context.getBean("handlerMapping", HandlerMapping.class);
                this.handlerMappings = Collections.singletonList(hm1);
            } catch (NoSuchBeanDefinitionException var3) {
                ;
            }
        }
        //如果還是沒(méi)有找到hadlerMapping就需要設(shè)定默認(rèn)的handlerMappings了
        if(this.handlerMappings == null) {
            this.handlerMappings = this.getDefaultStrategies(context, HandlerMapping.class);
            if(this.logger.isTraceEnabled()) {
                this.logger.trace("No HandlerMappings declared for servlet \'" + this.getServletName() + "\': using default strategies from DispatcherServlet.properties");
            }
        }
    }

下面是用debugger看看它究竟獲取了那些handlerMapping,如下:7個(gè)handlerMapping

除了這個(gè)初始化handlerMapping的initHandlerMapping方法,當(dāng)然還初始化了很多東西,如支持國(guó)際化的LocalResolver以及視圖生成的ViewResolver等的初始化過(guò)程,其余的有興趣自己跟著看一下,這里就不細(xì)跟了。到這里我們就知道了整個(gè)DispatcherServlet的初始化的大體流程了。

到此這篇關(guān)于SpringMVC中的DispatcherServlet初始化流程詳解的文章就介紹到這了,更多相關(guān)DispatcherServlet初始化流程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot+EasyPoi實(shí)現(xiàn)excel導(dǎo)出功能

    SpringBoot+EasyPoi實(shí)現(xiàn)excel導(dǎo)出功能

    最新小編遇到這樣一個(gè)需求,根據(jù)檢索條件查詢(xún)列表并將結(jié)果導(dǎo)出到excel,實(shí)現(xiàn)過(guò)程也非常簡(jiǎn)單,感興趣的朋友跟隨小編一起看看吧
    2021-09-09
  • Java泛型的使用限制實(shí)例分析

    Java泛型的使用限制實(shí)例分析

    這篇文章主要介紹了Java泛型的使用限制,結(jié)合實(shí)例形式分析了不能使用java泛型的情況以及泛型使用的相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2019-08-08
  • jconsole使用介紹(圖文)

    jconsole使用介紹(圖文)

    大家在學(xué)習(xí)java的時(shí)候,難免會(huì)對(duì)jvm進(jìn)行一些深入的了解。推薦大家使用jdk下面的jconsole.exe來(lái)輔助理解jvm的一些概念
    2015-12-12
  • Java中利用POI優(yōu)雅的導(dǎo)出Excel文件詳解

    Java中利用POI優(yōu)雅的導(dǎo)出Excel文件詳解

    這篇文章主要給大家介紹了關(guān)于Java中如何利用POI優(yōu)雅的導(dǎo)出Excel文件的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05
  • Java的NIO之通道channel詳解

    Java的NIO之通道channel詳解

    這篇文章主要介紹了Java的NIO之通道channel詳解,通道channel由java.nio.channels 包定義的,Channel 表示IO源與目標(biāo)打開(kāi)的連接,Channel類(lèi)類(lèi)似于傳統(tǒng)的"流",只不過(guò)Channel本身不能直接訪問(wèn)數(shù)據(jù),Channel只能與Buffer進(jìn)行交互,需要的朋友可以參考下
    2023-10-10
  • 解決springboot 實(shí)體類(lèi)String轉(zhuǎn)Date類(lèi)型的坑

    解決springboot 實(shí)體類(lèi)String轉(zhuǎn)Date類(lèi)型的坑

    這篇文章主要介紹了解決springboot 實(shí)體類(lèi)String轉(zhuǎn)Date類(lèi)型的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • SpringBoot實(shí)現(xiàn)在webapp下直接訪問(wèn)html,jsp

    SpringBoot實(shí)現(xiàn)在webapp下直接訪問(wèn)html,jsp

    這篇文章主要介紹了SpringBoot實(shí)現(xiàn)在webapp下直接訪問(wèn)html,jsp問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • CMD運(yùn)行Intellij Idea編譯后的class文件操作

    CMD運(yùn)行Intellij Idea編譯后的class文件操作

    這篇文章主要介紹了CMD運(yùn)行Intellij Idea編譯后的class文件操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-02-02
  • java數(shù)據(jù)結(jié)構(gòu)與算法之冒泡排序詳解

    java數(shù)據(jù)結(jié)構(gòu)與算法之冒泡排序詳解

    這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)與算法之冒泡排序,結(jié)合實(shí)例形式詳細(xì)分析了java冒泡排序的原理、實(shí)現(xiàn)技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2017-05-05
  • SpringBoot Filter修改返回內(nèi)容,解決請(qǐng)求卡死200的錯(cuò)誤

    SpringBoot Filter修改返回內(nèi)容,解決請(qǐng)求卡死200的錯(cuò)誤

    這篇文章主要介紹了SpringBoot Filter修改返回內(nèi)容,解決請(qǐng)求卡死200的錯(cuò)誤問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07

最新評(píng)論