SpringMvc @InitBinder

来源:https://www.cnblogs.com/lvbinbin2yujie/archive/2019/03/06/10459303.html
-Advertisement-
Play Games

這篇博客記錄@InitBinder怎麼起作用、起什麼作用? 首先,該註解被解析的時機,是該匹配Controller的請求執行映射的方法之前; 同時 @InitBinder標註的方法執行是多次的,一次請求來就執行一次。 當某個Controller上的第一次請求由SpringMvc前端控制器匹配到該Co ...


這篇博客記錄@InitBinder怎麼起作用、起什麼作用?

  首先,該註解被解析的時機,是該匹配Controller的請求執行映射的方法之前; 同時 @InitBinder標註的方法執行是多次的,一次請求來就執行一次。

  當某個Controller上的第一次請求由SpringMvc前端控制器匹配到該Controller之後,根據Controller的 class 類型 查找 所有方法上標註了@InitBinder的方法,並且存入RequestMappingHandlerAdapter的 initBinderCache,下次一請求執行對應業務方法之前時,可以走initBinderCache緩存,而不用再去解析@InitBinder; 所以 initBinder是controller級別的,一個controller實例中的所有@initBinder 只對該controller有效;

 

功能一.註冊Controller級別的 MVC屬性編輯器 (屬性編輯器功能就是將Web請求中的屬性轉成我們需要的類型) 

  @InitBinder唯一的一個屬性value,作用是限制對哪些 @RequestMapping 方法起作用,具體篩選條件就是通過@RequestMapping方法入參來篩選,預設不寫就代表對所有@RequestMapping的方法起作用;

  @InitBinder標註的方法, 方法入參和 @RequestMapping方法入參可選範圍一樣(這裡指的是比如HttpServletRequestModelMap這些), 通常一個入參 WebDataBinder 就夠我們使用了; @InitBinder標註的方法返回值, 必須為null,這裡我理解的是運行期的返回值;如果運行時返回值不為null,拋出異常 “@InitBinder methods should return void:”,編譯時IDEA會提示@InitBinder應該返回null,但是不影響編譯通過;

@InitBinder
    public  void initBinder(WebDataBinder binder, HttpServletRequest request){
        System.out.println(request.getParameter("date"));
        binder.registerCustomEditor(Date.class,new CustomDateEditor(new SimpleDateFormat("MM-dd-yyyy"),false));
    }

  上面是一個@InitBinder的簡單用法, 其中binder.registerCustomEditor(Date.class,new CustomDateEditor(new SimpleDateFormat("MM-dd-yyyy"),false)); 這樣一句話,作用就是將 自定義的MVC屬性編輯器PropertyEditor 註冊到當前binder的typeConverter的customEditors集合中,每一次請求和後端交互,每一個Controller方法入參都會創建一個Binder對象,binder對象相當於來完成請求和後端之間參數類型轉換的職能類;  註意,每次請求都會創建新的binder對象,就是說上次請求的customEditors不可復用 , 每次執行都會添加到當前方法入參交互的binder的customEditors中,而且每次執行真正請求方法之前,會把 匹配上的@InitBinder標註的方法執行一遍才開始處理;

  當請求中參數和方法入參開始進行轉換的時候,都會先使用自定義註冊的PropertyEditor,會首先根據需要的類型去binder的typeConverter的typeConverterDelegate的propertyEditorRegistry的cutomEditors集合中查找,有點繞,先記錄下,typeConverterDelegate的propertyEditorRegistry就是typeConverter對象本身, 所以就是去typeConverter對象的cutomEditors尋找自定義註冊的屬性編輯器,又回到了原點。 比如去customEditors中根據key為Date.class查找editor,  分為兩種情況,一種是找到了合適的屬性編輯器,調用其setValue、setAsText方法, 之後使用getValue就得到轉換後的值,  得到了轉換後的值,可能不是我們想要的類型,這時候就會使用 conversionService 重新來過,放棄之前的轉換;  是我們想要的類型就直接返迴轉換後的值;  第二種情況是沒找到合適的屬性編輯器, 直接調用 ConversionService 進行轉換工作;

上面方式是@InitBinder作為Controller級別的 SpringMvc屬性編輯器,  下麵記錄一下全局級別(所有@Controller)的屬性編輯器;

Xml中 註冊全局屬性編輯器到 ConfigurableWebBindingInitializer上,再將其註冊到 RequestMappingHandlerAdapter里;記錄原因,綁定binder的屬性編輯器時候,會將當前的 

Initializer中的屬性編輯器也給註冊到binder中,這樣就能實現全局的屬性編輯器

<!--<mvc:annotation-driven/>-->
    <!--取消註解驅動的話Spring4.3就要手動註冊RequestMappingHandlerMapping、RequestMappingHandlerAdapter-->

<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping">

</bean>
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
    <property name="webBindingInitializer" ref="initializer1"/>
</bean>

<bean id="initializer1" class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
    <property name="propertyEditorRegistrars">
       <list>
            <bean class="demo2.MyPropertyEditor"/>
       </list>
    </property>
</bean>

MyPropertyEditor.java

public class MyPropertyEditor implements PropertyEditorRegistrar {

    @Override
    public void registerCustomEditors(PropertyEditorRegistry registry) {
        registry.registerCustomEditor(Date.class,new MyDateEditor());
    }

    public static class MyDateEditor extends PropertyEditorSupport {
        @Override
        public void setValue(Object value) {
            super.setValue(value);
        }

        @Override
        public void setAsText(String text) throws IllegalArgumentException {
            Date d=null;
            try {
                System.out.println("我調用自己的全局MVC屬性編輯器");
                d=new SimpleDateFormat("MM-dd-yyyy").parse(text);
                setValue(d);
            } catch (ParseException e) {
                e.printStackTrace();
            }
        }
    }
}

上面兩段代碼就可以註冊 自定義的屬性編輯器到 所有@Controller中,相當於之前每個 Controller都使用了 @InitBinder;  但是這樣寫不太友好,SpringMvc<mvc:annotation-driven/>替我們註冊的很多東西可能就沒法使用了,意義不大,所以簡單改造了一下: 在之前記錄的Spring載入初始化容器的流程基礎上改造了下;

 

image

spring  XML文件仍然使用註解驅動:

<mvc:annotation-driven/>
<bean id="globalBeanDefinitionRegistry" class="demo2.GlobalBeanDefinitionRegistry">
    <property name="editorRegistrars">
        <list>
            <bean class="demo2.MyPropertyEditor"/>
        </list>
    </property>
</bean>

 

自定義的 GlobalBeanDefinitionRegistry代碼如下:  簡單說下原理,在Spring原有註解驅動的基礎上,改變了webBindingInitializer,使它可以自由地配置方式添加屬性編輯器;優點就是,不破壞SpringMvc註解驅動帶給我們的好處,可以自定義添加屬性全局的編輯器;缺點就是 代碼中判斷邏輯的 處理器映射器適配器 RequestMappingHandlerAdapter是硬編碼的,Spring4可能還好用,Spring3突然又不支持了,同樣也是有解決方案的

用法其實就是在Spring初始化容器中對象之前移花接木地替換我們的 webBindingInitalizer.

public class GlobalBeanDefinitionRegistry  implements BeanDefinitionRegistryPostProcessor {
        private PropertyEditorRegistrar[]  editorRegistrars;
    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
        if(registry.containsBeanDefinition("org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter")){
            BeanDefinition beanDefinition = registry.getBeanDefinition("org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter");
            PropertyValue pv = beanDefinition.getPropertyValues().getPropertyValue("webBindingInitializer");
            BeanDefinition intializer= (BeanDefinition) pv.getValue();
            intializer.getPropertyValues().addPropertyValue("propertyEditorRegistrars",editorRegistrars);
        }
    }

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {

    }
    public void setEditorRegistrars(PropertyEditorRegistrar[] editorRegistrars) {
        this.editorRegistrars = editorRegistrars;
    }
}

因為@InitBinder方法 作為Controller級別的屬性編輯器 和全局的自定義Mvc屬性編輯器沒有太大差別,所以下麵講一些別的用法:

 

功能二. WebDatabinder的setFieldDefaultPrefix(String fieldDefaultPrefix)

作用:將SpringMvc請求參數帶有fieldDefaultPrefix的參數,去掉該首碼再綁定到對應請求入參上

  想來想去,也沒搞明白這個方法的意義,以及實際用途,想到一種實際中可能出現的情況,覺得是個有幾率出現的事情,正好可以用該方法可以解決;

問題:假設 後臺 用兩個對象來接受請求參數(SpringMvc可以做到),Pojo、Pojo2對象,他們兩個屬性如下:發現兩個對象都有個name屬性,問題來了,前臺我們不能傳兩個 name屬性吧,那樣接收肯定會出錯(我這裡沒嘗試過),原有對象不修改的基礎上可行方案如下:

@Setter
@Getter
@ToString  // 代碼整潔所以使用lombok,可以自行百度
public class Pojo {
    private String name;
    private String haircolor;
}

@Setter
@ToString
public class Pojo2 {
    private String name;
    private int age;
}

 

在兩個對象上各使用@ModelAttribute,來給對象分別起別名, @Initbinder這裡value屬性指定 別名,然後給不同的參數加上了首碼 person. 、cat.  ;註意這裡的兩個 . 號 

 @RequestMapping("/test3")
    @ResponseBody
    public String test3(@ModelAttribute("person") Pojo person, @ModelAttribute("cat") Pojo2 cat){
        return "test Response Ok!"+person+","+cat;
    }

    @InitBinder("person")
    public void initPerson(WebDataBinder binder){
          binder.setFieldDefaultPrefix("person.");
    }

    @InitBinder("cat")
    public void initCat(WebDataBinder binder){
        binder.setFieldDefaultPrefix("cat.");
    }

 

這樣請求URL:………… test3?person.name=lvbinbin&cat.name=xiaobinggan&haircolor=black&age=20 這裡前臺改動的地方就是 person.name和 cat.name,而其他獨有屬性不需要首碼也可以對應賦給pojo、pojo2;  補充說明,@ModelAttribute註解不可以省略,通過這個取的別名來決定哪個@InitBinder對其生效

查看效果圖:

image

 

簡單記錄下,因為這個 defaultPrefix 所在代碼確實不好找:

protected void doBind(MutablePropertyValues mpvs) {
		checkFieldDefaults(mpvs);    //這裡就是 defaultPrefix生效的地方
		checkFieldMarkers(mpvs);
		super.doBind(mpvs);
	}
//效果就是請求中包含defaultPrefix的,將其首碼去掉保存
protected void checkFieldDefaults(MutablePropertyValues mpvs) {
	if (getFieldDefaultPrefix() != null) {
		String fieldDefaultPrefix = getFieldDefaultPrefix();
		PropertyValue[] pvArray = mpvs.getPropertyValues();
		for (PropertyValue pv : pvArray) {
			if (pv.getName().startsWith(fieldDefaultPrefix)) {
				String field = pv.getName().substring(fieldDefaultPrefix.length());
				if (getPropertyAccessor().isWritableProperty(field) && !mpvs.contains(field)) {
					mpvs.add(field, pv.getValue());
				}
				mpvs.removePropertyValue(pv);
			}
		}
	}
}

 

功能三.WebDataBinder的setDisallowedFields(String ….disallowedFields);

作用:SpringMvc接收請求參數時候,有些參數禁止的,不想接收,我也沒遇到過啥情況禁止接收參數,這時候可以設置setDisallowedFields不接受參數

 @RequestMapping("/test4")
    @ResponseBody
    public String test4(@ModelAttribute("pojo2") Pojo2 pojo){
        return "test Response Ok!"+pojo;
    }

    @InitBinder("pojo2")
    public void disallowFlied(WebDataBinder binder){
        binder.setDisallowedFields("age");
    }

 

簡單貼下效果圖:

image

 

正好發現了日誌輸出證明這一點:

image

那就順便把代碼貼一下,萬一要用呢;  另外補充一下,disallowedFields支持 * 通配符;

protected void checkAllowedFields(MutablePropertyValues mpvs) {
		PropertyValue[] pvs = mpvs.getPropertyValues();
		for (PropertyValue pv : pvs) {
			String field = PropertyAccessorUtils.canonicalPropertyName(pv.getName());
			if (!isAllowed(field)) {
				mpvs.removePropertyValue(pv);
				getBindingResult().recordSuppressedField(field);
				if (logger.isDebugEnabled()) {
					logger.debug("Field [" + field + "] has been removed from PropertyValues " +
							"and will not be bound, because it has not been found in the list of allowed fields");
				}
			}
		}
	}
protected boolean isAllowed(String field) {
	String[] allowed = getAllowedFields();
	String[] disallowed = getDisallowedFields();
	return ((ObjectUtils.isEmpty(allowed) || PatternMatchUtils.simpleMatch(allowed, field)) &&
			(ObjectUtils.isEmpty(disallowed) || !PatternMatchUtils.simpleMatch(disallowed, field)));
}

 

@Initbinder的用法簡而言之,就是Controller級別的屬性編輯器,將請求中的String類型轉為我們需要的參數,但是從效率、記憶體分析,感覺一直在創建新的屬性編輯器集合,如果屬性編輯器太多是不是會占用大量記憶體呢,那請求達到一定多的數量,這個對象是不是太多了呢?


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 中文編程 "知乎專欄" 原文 "地址" 基本參考https://pragprog.com/book/tpantlr2/the definitive antlr 4 reference 一書"Building a Calculator Using a Visitor"一節, 僅添加了數學乘除法符號的支 ...
  • 在可預見的未來, 高考仍是最重要的也最有社會影響力的人才選拔機制. 很久沒有關註, 最近得知高考自選科目中開始增加了編程一項(見 "如何評價2017浙江高考七選三科目中包含技術?" ). 雖然個人對編程是否應該進入高考仍有保留看法, 但至少全民(都應該可以)編程這一趨勢已經很明顯了. 這應該是中文編 ...
  • 筆者環境 centos7 python3 pytesseract只是tesseract-ocr的一種實現介面。所以要先安裝tesseract-ocr(大名鼎鼎的開源的OCR識別引擎)。 依賴安裝 安裝依賴的leptonica庫 安裝tesseract-ocr 安裝語言包: 安裝pytesseract ...
  • Maven 有一個生命周期,當你運行 mvn install 的時候被調用。這條命令告訴 Maven 執行一系列的有序的步驟,直到到達你指定的生命周期。遍歷生命周期旅途中的一個影響就是,Maven 運行了許多預設的插件目標,這些目標完成了像編譯和創建一個 JAR 文件這樣的工作。 一個jar包,會有 ...
  • 為什麼要談這個topic? 實踐中,質量保障體系的建設,主要針對兩個目標: 一是不斷提高目標業務測試覆蓋率,保障面向客戶的產品質量;二就是儘可能的提高人效,增強迭代效率。而構建全鏈路質量卡點就是整個體系建設的核心手段。筆者用下圖來描述這整個鏈路: 可以看到,雖然保障業務迭代的方向性正確排在最前面,但 ...
  • 真的是忙頭暈了,學業、ACM打題、班級活動、自學新東西,哇這充實的大學~ L1-007 念數字 輸入一個整數,輸出每個數字對應的拼音。當整數為負數時,先輸出fu字。十個數字對應的拼音如下: 0: ling 1: yi 2: er 3: san 4: si 5: wu 6: liu 7: qi 8: ...
  • 首先創建一個test類: 在main方法里寫上如下代碼: 在工程目錄下新建一個generator.xml文件: 最後的table標簽是自己資料庫中表的名字;資料庫的連接信息需要自己修改 執行test類就會自動生成自己以上設置table標簽中數據中表的對應的實體類,dao層介面以及對應的mapper映 ...
  • 可變對象與不可變對象 要理解深拷貝和淺拷貝,首先要理解可變對象和不可變對象。 不可變對象:該對象所指向的記憶體中的值不能被改變,修改對象的值時,由於其指向的值不能被改變,因此實際上是在記憶體中重新開闢一個地址用來存儲新的值,然後將對象指向這個新值。本質上是兩個對象,賦值前後對象id發生了變化。pytho ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...