1. 程式人生 > >Spring mvc請求處理流程詳解(一)之檢視解析

Spring mvc請求處理流程詳解(一)之檢視解析

前言

  Spring mvc框架相信很多人都很熟悉了,關於這方面的資料也是一搜一大把。但是感覺講的都不是很細緻,讓很多初學者都雲裡霧裡的。本人也是這樣,之前研究過,但是後面一段時間不用發現又忘記了。所以決定寫下來,以備後用。
  本系列文基於spring-4.3.1,配置方式全部基於java-based方式

從配置講起

先上一段配置的程式碼:

@EnableWebMvc
@Configuration
public class MvcConfig extends WebMvcConfigurerAdapter {

    @Override
    public void configureDefaultServletHandling
(DefaultServletHandlerConfigurer configurer) { configurer.enable(); } @Override public void configureViewResolvers(ViewResolverRegistry registry) { registry.jsp("/WEB-INF/jsp/", ".jsp"); registry.enableContentNegotiation(new MappingJackson2JsonView()); } @Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) { configurer.favorPathExtension(true) .ignoreAcceptHeader(true) .parameterName("mediaType") .defaultContentType(MediaType.TEXT_HTML) .mediaType("html"
, MediaType.TEXT_HTML) .mediaType("json", MediaType.APPLICATION_JSON); } @Bean(name = "multipartResolver") // 檔案上傳bean public CommonsMultipartResolver commonsMultipartResolver() { return new CommonsMultipartResolver(); } }

  基於java-based方式的spring mvc配置,需要建立一個配置類並實現WebMvcConfigurer 介面,WebMvcConfigurerAdapter 抽象類是對WebMvcConfigurer 介面的簡單抽象(增加了一些預設實現),所以上面配置程式碼選擇直接繼承WebMvcConfigurerAdapter 。然後根據專案的需要實現介面中特定的方法,最後要注意的是,要在配置類上標註@EnableWebMvc
  到這裡可能有人會問,我怎麼知道實現哪些方法?具體該怎麼配?它們之間的處理流程是怎樣的?好的,別急,我們一步步來。
  首先第一步,我們需要知道WebMvcConfigurer 介面都提供了哪些回撥方法?

WebMvcConfigurer

package org.springframework.web.servlet.config.annotation;
/**
 * 篇幅原因,我們先只介紹Spring mvc常用的一些方法
 */
public interface WebMvcConfigurer {

    void addFormatters(FormatterRegistry registry);

    void configureMessageConverters(List<HttpMessageConverter<?>> converters);

    void extendMessageConverters(List<HttpMessageConverter<?>> converters);

    Validator getValidator();

    /* 配置內容裁決的一些選項*/
    void configureContentNegotiation(ContentNegotiationConfigurer configurer);

    void configureAsyncSupport(AsyncSupportConfigurer configurer);

    /* @since 4.0.3 */
    void configurePathMatch(PathMatchConfigurer configurer);

    /*引數解析*/
    void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers);

    /*返回值解析*/
    void addReturnValueHandlers(List<HandlerMethodReturnValueHandler> returnValueHandlers);

    /*異常處理*/
    void configureHandlerExceptionResolvers(List<HandlerExceptionResolver> exceptionResolvers);

    void extendHandlerExceptionResolvers(List<HandlerExceptionResolver> exceptionResolvers);

    void addInterceptors(InterceptorRegistry registry);

    MessageCodesResolver getMessageCodesResolver();

    void addViewControllers(ViewControllerRegistry registry);

    /**
     * 這裡配置檢視解析器
     */
    void configureViewResolvers(ViewResolverRegistry registry);

    /**
     *靜態資源處理
     */
    void addResourceHandlers(ResourceHandlerRegistry registry);

    void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer);

    void addCorsMappings(CorsRegistry registry);
}

下面我們開始著重講解以下幾個常用的方法:

void configureViewResolvers(ViewResolverRegistry registry);
void configureContentNegotiation(ContentNegotiationConfigurer configurer);
void addViewControllers(ViewControllerRegistry registry);
void addResourceHandlers(ResourceHandlerRegistry registry);
void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer);

1. configureViewResolvers(ViewResolverRegistry registry)

  從方法名稱我們就能看出這個方法是用來配置檢視解析器的,該方法的引數ViewResolverRegistry 是一個註冊器,用來註冊你想自定義的檢視解析器等。ViewResolverRegistry 常用的幾個方法:

1).enableContentNegotiation
/** 啟用內容裁決檢視解析器*/
public void enableContentNegotiation(View... defaultViews) {
        initContentNegotiatingViewResolver(defaultViews);
    }

  該方法會建立一個內容裁決解析器ContentNegotiatingViewResolver ,該解析器不進行具體檢視的解析,而是管理你註冊的所有檢視解析器,所有的檢視會先經過它進行解析,然後由它來決定具體使用哪個解析器進行解析。具體的對映規則是根據請求的media types來決定的。

2).  UrlBasedViewResolverRegistration
    public UrlBasedViewResolverRegistration jsp(String prefix, String suffix) {
        InternalResourceViewResolver resolver = new InternalResourceViewResolver();
        resolver.setPrefix(prefix);
        resolver.setSuffix(suffix);
        this.viewResolvers.add(resolver);
        return new UrlBasedViewResolverRegistration(resolver);
    }

  該方法會註冊一個內部資源檢視解析器InternalResourceViewResolver 顯然訪問的所有jsp都是它進行解析的。該方法引數用來指定路徑的字首和檔案字尾,如:  

    registry.jsp("/WEB-INF/jsp/", ".jsp");

  對於以上配置,假如返回的檢視名稱是example,它會返回/WEB-INF/jsp/example.jsp給前端,找不到則報404。
  

3).  beanName
    public void beanName() {
        BeanNameViewResolver resolver = new BeanNameViewResolver();
        this.viewResolvers.add(resolver);
    }

  該方法會註冊一個BeanNameViewResolver 檢視解析器,這個解析器是幹嘛的呢?它主要是將檢視名稱解析成對應的bean。什麼意思呢?假如返回的檢視名稱是example,它會到spring容器中找有沒有一個叫example的bean,並且這個bean是View.class型別的?如果有,返回這個bean。
  

4).  viewResolver
    public void viewResolver(ViewResolver viewResolver) {
        if (viewResolver instanceof ContentNegotiatingViewResolver) {
            throw new BeanInitializationException(
                    "addViewResolver cannot be used to configure a ContentNegotiatingViewResolver. Please use the method enableContentNegotiation instead.");
        }
        this.viewResolvers.add(viewResolver);
    }

  這個方法想必看名字就知道了,它就是用來註冊各種各樣的檢視解析器的,包括自己定義的。
  

2. configureContentNegotiation(ContentNegotiationConfigurer configurer)

  上面一節我們講了configureViewResolvers 方法,假如在該方法中我們啟用了內容裁決解析器,那麼configureContentNegotiation(ContentNegotiationConfigurer configurer) 這個方法是專門用來配置內容裁決的一些引數的。這個比較簡單,我們直接通過一個例子看:
  

   public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
       /* 是否通過請求Url的副檔名來決定media type */
        configurer.favorPathExtension(true)  
                 /* 不檢查Accept請求頭 */
                .ignoreAcceptHeader(true)
                .parameterName("mediaType")
                 /* 設定預設的media yype */
                .defaultContentType(MediaType.TEXT_HTML)
                 /* 請求以.html結尾的會被當成MediaType.TEXT_HTML*/
                .mediaType("html", MediaType.TEXT_HTML)
                /* 請求以.json結尾的會被當成MediaType.APPLICATION_JSON*/
                .mediaType("json", MediaType.APPLICATION_JSON);
    }

 到這裡我們就可以舉個例子來進一步熟悉下我們上面講的知識了,假如我們MVC的配置如下:

    @EnableWebMvc
    @Configuration
    public class MvcConfig extends WebMvcConfigurerAdapter {

        @Override
        public void configureViewResolvers(ViewResolverRegistry registry) {
            registry.jsp("/WEB-INF/jsp/", ".jsp");
            registry.enableContentNegotiation(new MappingJackson2JsonView());
        }

        @Override
        public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
            configurer.favorPathExtension(true)
                    .ignoreAcceptHeader(true)
                    .parameterName("mediaType")
                    .defaultContentType(MediaType.TEXT_HTML)
                    .mediaType("html", MediaType.TEXT_HTML)
                    .mediaType("json", MediaType.APPLICATION_JSON);
        }
    }

  controller的程式碼如下:

    @Controller
    public class ExampleController {
         @RequestMapping("/example1")
         public ModelAndView example1() {
            Map<String, String> map = new HashMap();
            map.put("1", "a");
            map.put("2", "b");
            return new ModelAndView("example1", map);
        }
    }

  在WEB-INF/jsp目錄下建立一個example1.jsp檔案,內容隨意。現在啟動tomcat,在瀏覽器輸入以下連結:http://localhost:8080/example1.json,瀏覽器返回如下:
  這裡寫圖片描述
在瀏覽器輸入http://localhost:8080/example1 或者http://localhost:8080/example1.html,返回如下:
這裡寫圖片描述
  顯然,兩次使用了不同的檢視解析器,那麼底層到底發生了什麼?在配置裡我們註冊了兩個檢視解析器:ContentNegotiatingViewResolverInternalResourceViewResolver,還有一個預設檢視:MappingJackson2JsonView。controller執行完畢之後返回一個ModelAndView,其中檢視的名稱為example1。返回首先會交給ContentNegotiatingViewResolver 進行檢視解析處理,而ContentNegotiatingViewResolver 會先把檢視名example1交給它持有的所有ViewResolver嘗試進行解析(本例項中只有InternalResourceViewResolver),然後根據請求的mediaType,再將example1.mediaType(這裡是example1.json 和example1.html)作為檢視名讓所有檢視解析器解析一遍,兩步解析完畢之後會獲得一堆候選的List<View> 再加上預設的MappingJackson2JsonView ,最後根據請求的media type從候選的List<View> 中選擇一個最佳的返回,至此檢視解析完畢。現在就可以理解上例中為何請求連結加上.json 和不.json 結果會不一樣。當加上.json 時,表示請求的media type 為MediaType.APPLICATION_JSON,而InternalResourceViewResolver 解析出來的檢視的ContentType與其不符,而與MappingJackson2JsonView 的ContentType相符,所以選擇了MappingJackson2JsonView 作為檢視返回。當不加.json 請求時,預設的media type 為MediaType.TEXT_HTML,所以就使用了InternalResourceViewResolver解析出來的檢視作為返回值了。我想看到這裡你已經大致可以自定義檢視了。

3. addViewControllers(ViewControllerRegistry registry)

  此方法可以很方便的實現一個請求到檢視的對映,而無需書寫controller,例如:
  

    @Override  
    public void addViewControllers(ViewControllerRegistry registry){  
        registry.addViewController("/login").setViewName("login");  
    } 

  這是訪問${domain}/login時,會直接返回login頁面。

4. addResourceHandlers(ResourceHandlerRegistry registry)

  此方法用來專門註冊一個Handler,來處理靜態資源的,例如:圖片,js,css等。舉例:
  

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/resource/**").addResourceLocations("/WEB-INF/static/");
    }

  當你請求http://localhost:8083/resource/1.png時,會把/WEB-INF/static/1.png返回。注意:這裡的靜態資源是放置在WEB-INF目錄下的。

5. configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer)

  用法:

    @Override
    public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
        configurer.enable();
    }

  此時會註冊一個預設的Handler:DefaultServletHttpRequestHandler,這個Handler也是用來處理靜態檔案的,它會嘗試對映/*。當DispatcherServelt對映/時(//* 是有區別的),並且沒有找到合適的Handler來處理請求時,就會交給DefaultServletHttpRequestHandler 來處理。注意:這裡的靜態資源是放置在web根目錄下,而非WEB-INF 下。
  可能這裡的描述有點不好懂(我自己也這麼覺得),所以簡單舉個例子,例如:在webroot目錄下有一個圖片:1.png 我們知道Servelt規範中web根目錄(webroot)下的檔案可以直接訪問的,但是由於DispatcherServlet配置了對映路徑是:/ ,它幾乎把所有的請求都攔截了,從而導致1.png 訪問不到,這時註冊一個DefaultServletHttpRequestHandler 就可以解決這個問題。其實可以理解為DispatcherServlet破壞了Servlet的一個特性(根目錄下的檔案可以直接訪問),DefaultServletHttpRequestHandler是幫助迴歸這個特性的。

題外話

  問: //* 有什麼區別?
  答: /會攔截除了jsp以外的所有url,/* 會攔截所有url,包括jsp。例如:在webroot下面有一個test.jsp,當DispatcherServlet 配置對映/ 時,瀏覽器輸入:http://localhost:8083/test.jsp 這個jsp是可以直接訪問的,並且不經過DispatcherServlet ,而當DispatcherServlet 配置對映/* 時,這個請求就會被DispatcherServlet 攔截。

相關推薦

Spring mvc請求處理流程檢視解析

前言   Spring mvc框架相信很多人都很熟悉了,關於這方面的資料也是一搜一大把。但是感覺講的都不是很細緻,讓很多初學者都雲裡霧裡的。本人也是這樣,之前研究過,但是後面一段時間不用發現又忘記了。所以決定寫下來,以備後用。   本系列文基於spring-

Spring Boot啟動流程

轉載:http://www.cnblogs.com/xinzhao/p/5551828.html 環境 本文基於Spring Boot版本1.3.3, 使用了spring-boot-starter-web。 配置完成後,編寫了程式碼如下: @

、S5PV210的啟動流程

210整個啟動流程可以大致分為三個階段,分別為:      1.執行IROM中的程式碼       2.執行UBOOT的BL1       3.執行UBOOT的BL2,最後啟動核心 IROM是2

MVC HTML 幫助器

@Html.BeginFrom()                                                                                                                                         

c/c++預處理過程條件編譯及預定義的巨集

未經博主同意不得私自轉載!不準各種形式的貼上複製本文及盜圖! 首先對於上篇文章中巨集定義的補充: (1)#define NAME"zhangyuncong" 程式中有"NAME"則,它會不會被替換呢? (2)#define 0x abcd 可以嗎?也就是說,可不可以用不是

freescale飛思卡爾 HC9S12 系列微控制器 Flash擦寫時鐘設定

       Flash擦寫的內容,個人做HC9S12系列微控制器時覺得應該是各模組內容中最難而且是最麻煩的一步了。只有能夠對Flash進行擦寫以後,所做的Bootloader才有真正手段將串列埠或者其他通訊手段接收到的資料或者程式寫入Flash中進行程式或者資料的更新。當初做Flash的擦寫也遇到了很多問題

itop4412 LCD裝置驅動DEVICE

LCD的工作,在kernel中有device和driver兩個描述,這也是必然。 一.先看device 在palt-s5p/dev-fimd-s5p.c 定義了一個 struct platform_device s3c_device_fb 平臺裝置

Spring MVC請求處理流程及架構

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; H

Spring MVC請求處理流程

從web.xml中 servlet的配置開始, 根據servlet攔截的url-parttern,來進行請求轉發 Spring MVC工作流程圖 圖一 圖二  Spring工作流程描述       1. 使用者向伺服器傳送請求,請求被Spring 前端控制Servelt Di

spring mvc請求處理流程/原理

1.spring mvc請所有的請求都提交給DispatcherServlet,它會委託應用系統的其他模組負責負責對請求進行真正的處理工作。2.DispatcherServlet查詢一個或多個Handl

Spring MVC請求處理流程分析

一、簡介 Spring MVC框架在工作中經常用到,配置簡單,使用起來也很方便,很多書籍和部落格都有介紹其處理流程,但是,對於

spring @Transactional註解參數13

基於接口 ack -a 事物 null span ports readonly 可見度 事物註解方式: @Transactional 當標於類前時, 標示類中所有方法都進行事物處理 , 例子: 1 @Transactional public class TestServ

安卓專案實戰強大的網路請求框架okGo使用:實現get,post基本網路請求,下載上傳進度監聽以及對Callback自定義的深入理解

1.新增依賴 //必須使用 compile 'com.lzy.net:okgo:3.0.4' //以下三個選擇新增,okrx和okrx2不能同時使用,一般選擇新增最新的rx2支援即可 compile 'com.lzy.net:okrx:1.0.2' compile 'com.lzy

Spring------概述

目錄 1、什麼是 Spring ? 2、Spring 起源 3、Spring 特點 4、Spring 框架結構 5、Spring 框架特徵  6、Spring 優點     本系列教程我們將對 Spring 進行詳解的介紹,

Consumer 拉取日誌流程

文章目錄 一、Consumer的poll模型 poll執行流程 相關原始碼解析 二、獲取要消費的partition 三種訂閱模式 AUTO_TOPICS 和 AUTO_PATTE

Spring ------- AOP

1. AOP 簡介 ​ AOP(Aspect Oriented Programming),通常稱為面向切面程式設計。它利用一種稱為"橫切"的技術,剖解開封裝的物件內部,並將那些影響了多個類的公共行為封裝到一個可重用模組,並將其命名為"Aspect",即切面。所謂"切面",簡單說就是那些與業務無關,卻為業務模

spring的BeanFactory和ApplicationContext原始碼

轉自http://www.sandzhang.com/blog/2011/04/10/Spring-BeanFactory-ApplicationContext-Detail-1/ 版本:spring-framework-3.0.5.RELEASE Spring的最核心的部分就是BeanFactory了,

(轉)Spring boot——logback.xml 配置

回到頂部1 根節點<configuration>包含的屬性scan:當此屬性設定為true時,配置檔案如果發生改變,將會被重新載入,預設值為true。scanPeriod:設定監測配置檔案是否有修改的時間間隔,如果沒有給出時間單位,預設單位是毫秒。當scan為true時,此屬性生效。預設的時間間隔

Spring boot——logback.xml 配置

原文地址:https://www.cnblogs.com/lixuwu/p/5810912.html                   https://aub.iteye.com/blog/1101260 閱

Spring:簡介

Spring Framework創始人:Rod Johnson. 計算機專業本科,音樂學博士。有著相當豐富的C/C++技術背景的Rod早在1996年就開始了對Java伺服器端技術的研究。 輪子理論推崇者: 輪子理論:不用重複發明輪子 IT 行業:直接使用寫好