Spring 源碼(4)在Spring配置文件中自定義標簽如何實現?

来源:https://www.cnblogs.com/redwinter/archive/2022/04/19/16165878.html
-Advertisement-
Play Games

Spring 配置文件自定義標簽的前置條件 在上一篇文章https://www.cnblogs.com/redwinter/p/16165274.html Spring BeanFactory的創建過程中瞭解了BeanDefinition的載入和BeanFactory的創建,並且提到了Spring留 ...


Spring 配置文件自定義標簽的前置條件

在上一篇文章https://www.cnblogs.com/redwinter/p/16165274.html Spring BeanFactory的創建過程中瞭解了BeanDefinition的載入和BeanFactory的創建,並且提到了Spring留了一個擴展點就是用戶可以自定義標簽進行解析BeanDefinition

基於Spring源碼在處理定製的標簽時是通過定製的命名空間處理器和xsd文件進行解析的,在springclasspath下的META-INF/spring.schemasMETA-INF/spring.handlers,並且需要將標簽的解析器註冊到BeanDefinition的解析器中,這樣說起來比較抽象,接下來我們自己定義一個標簽就明瞭了。

定義標簽屬性類

創建一個需要解析的標簽的屬性,比如在Spring配置文件中經常看到的<context:component-scan base-package="com.redwinter.test"/> ,這裡的component-scan就是標簽屬性。

/**
 * @author <a href="https://www.cnblogs.com/redwinter/">redwinter</a>
 * @since 1.0
 **/
public class Redwinter {

	private String username;
	private String email;
	private String password;

	public String getUsername() {
		return username;
	}

	public void setUsername(String username) {
		this.username = username;
	}

	public String getEmail() {
		return email;
	}

	public void setEmail(String email) {
		this.email = email;
	}

	public String getPassword() {
		return password;
	}

	public void setPassword(String password) {
		this.password = password;
	}
}

定義一個Redwinter類,裡面三個屬性,當然你可以自己定義你需要的屬性,我這裡就隨便寫啦。

定義標簽屬性解析器類

定義好標簽的屬性之後就需要定義一個解析器對這些屬性進行解析,定義解析器需要繼承AbstractSingleBeanDefinitionParser,這個類是實現了BeanDefinitionParser的類,他下麵有很多實現類,一般來說我們的Bean都是單例的,那就繼承AbstractSingleBeanDefinitionParser即可。

/**
 * @author <a href="https://www.cnblogs.com/redwinter/">redwinter</a>
 * @since 1.0
 **/
public class RedwinterBeanDefinitionParser extends AbstractSingleBeanDefinitionParser {

	@Override
	protected Class<?> getBeanClass(Element element) {
		return Redwinter.class;
	}

	@Override
	protected void doParse(Element element, BeanDefinitionBuilder builder) {
		/**
		 * 自定義解析xml的自定義欄位,並做相應的其他處理
		 */
		String username = element.getAttribute("username");
		String email = element.getAttribute("email");
		String password = element.getAttribute("password");
		if (StringUtils.hasText(username)){
			builder.addPropertyValue("username",username);
		}
		if (StringUtils.hasText(email)){
			builder.addPropertyValue("email",email);
		}
		if (StringUtils.hasText(password)){
			builder.addPropertyValue("password",password);
		}
	}
}

這個解析器主要是重寫了父類的兩個方法,一個是getBeanClass用於返回對應的標簽屬性類,一個是解析屬性doParser,這裡我只是從element中獲取出來然後進行了下判斷在加入到屬性值中,當然這裡你可以自定義自己的邏輯處理。

定義命名空間處理器類

定義命名空間處理器需要繼承NamespaceHandlerSupport,然後重寫他的init方法,將解析器註冊進去,這個解析器就是上面定義的用來解析標簽屬性的解析器。

/**
 * @author <a href="https://www.cnblogs.com/redwinter/">redwinter</a>
 * @since 1.0
 **/
public class RedwinterNameSpaceHandler extends NamespaceHandlerSupport {
	@Override
	public void init() {
		// 這裡的屬性必須和xsd中指定的屬性一致,否則報錯
		//org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Configuration problem: Cannot locate BeanDefinitionParser for element [dl]
		registerBeanDefinitionParser("dl",new RedwinterBeanDefinitionParser());
	}
}

這裡需要註意的是,進行註冊時需要指定一個elementName,這個值必須和xml中定義的名稱一致,否者的話就會報如下錯:

org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Configuration problem: Cannot locate BeanDefinitionParser for element [dl]

我這裡定義的元素名稱叫dl

定義xsd文件

xsd文件就是spring進行xml解析時解析的標簽,當然你可以定義dtd文件,不過現在一般都用xsd文件,我這裡命名為redwinter.xsd,完整文件如下:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>

<xsd:schema xmlns="http://www.redwinter.com/schema/redwinter"
			xmlns:xsd="http://www.w3.org/2001/XMLSchema"
			targetNamespace="http://www.redwinter.com/schema/redwinter"
			elementFormDefault="qualified"
			attributeFormDefault="unqualified">
	<xsd:element name="dl">
		<xsd:complexType>
			<xsd:attribute name="id" type="xsd:string"/>
			<xsd:attribute name="username" type="xsd:string" use="required"/>
			<xsd:attribute name="email" type="xsd:string" use="required"/>
			<xsd:attribute name="password" type="xsd:string" use="required"/>
		</xsd:complexType>
	</xsd:element>
</xsd:schema>

這裡有幾個點需要註意: schema標簽下有個targetNamespace,這裡指定了命名空間叫http://www.redwinter.com/schema/redwinter ,那麼在進行spring配置文件的時候引入的namespace就是這個,然後有個name="dl",這裡的這個dl就是處理器中定義的元素名稱,而且必須一致,不然解析不到,下麵定義的屬性就是標簽屬性類中定義的刷新,這個id是表示唯一的Bean名稱。

編寫spring.schemas和spring.handlers文件

這裡直接列出完整文件內容:

  • spring.schemas文件
http\://www.redwinter.com/schema/redwinter.xsd=META-INF/redwinter.xsd

這裡需要註意的是,這裡配置的key也是需要在spring配置文件中引入的,value就是上一步定義的xsd文件所在路徑

  • spring.handlers文件
http\://www.redwinter.com/schema/redwinter=com.redwinter.test.RedwinterNameSpaceHandler

這裡配置的key就是上一步定義的xsd文件中定義的targetNamespacevalue就是定義的命名空間處理器。

到這裡自定義標簽和解析就完成了,最後就需要在spring配置文件中引入並配置。

驗證自定義屬性標簽

我這裡定義個角spring-test.xml的文件進行配置

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"
	   xmlns:context="http://www.springframework.org/schema/context"
	   xmlns:redwinter="http://www.redwinter.com/schema/redwinter"
	   xsi:schemaLocation="http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
		http://www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans-3.1.xsd
		http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd
		http://www.redwinter.com/schema/redwinter  http://www.redwinter.com/schema/redwinter.xsd
		">
	<!--自定義標簽-->
	<redwinter:dl id ="redwinter" email="[email protected]" password="123456" username="冬玲記憶"/>
	<redwinter:dl id ="redwinter123456"  email="[email protected]" password="123456" username="冬玲記憶"/>

</beans>

驗證是否配置正確:

public class BeanCreate {

	@Test
	public void classPathXml() {
//		ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring-test.xml");
		ClassPathXmlApplicationContext context = new MyClassPathXmlApplicationContext("classpath:spring-test.xml");
      
		Redwinter redwinter = (Redwinter) context.getBean("redwinter");
		System.out.println(redwinter.getEmail());

		Redwinter redwinter123456 = (Redwinter) context.getBean("redwinter123456");
		System.out.println(redwinter123456.getEmail());
	}
}

輸出:

[email protected]
[email protected]

那說明自定義標簽生效了,並且成功解析出來。

接下來就是繼續介紹Spring 容器的實現AbstractApplicationContex#refresh的第三個方法,這個方法其實就是BeanFactory使用的前戲準備,而第一個方法是BeanFactory刷新的前戲準備。


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

-Advertisement-
Play Games
更多相關文章
  • 迭代器模式是什麼 迭代器模式是一種行為設計模式, 讓你能在不暴露集合底層表現形式 (列表、 棧和樹等) 的情況下遍歷集合中所有的元素。 為什麼用迭代器模式 當集合背後為複雜的數據結構, 且你希望對客戶端隱藏其複雜性時(出於使用便利性或安全性的考慮),可以使用迭代器模式。迭代器封裝了與複雜數據結構進行 ...
  • **導讀:**風控是金融最常見的場景之一,本文將從業務和技術架構兩個層面和大家探討如何落地智能風控中台系統。分享主要圍繞下麵五點展開: 風控中台的設計背景 策略的全周期管理 模型的全周期管理 業務架構和能力原子化 應用案例 -- 01 風控中台的設計背景 首先大風控體系或者風控中台的建設在本質上是服 ...
  • 在面向對象編程中,先編寫表示現實世界中的事物和情景的類,並基於這些類來創建對象。基於類創建對象時,每個對象都自動具備類的通用行為,同時可根據需要賦予每個對象獨特的個性,在實例中存儲特定信息及操作根據類來創建對象被稱為實例化類,也可以用來擴展既有類的功能,讓相似的類能夠高效地共用代碼 一、創建和使用類 ...
  • 1.java的"一次編寫,處處運行"如何實現?: 答:java之所有能實現一次編譯,到處運行,是因為java在每個系統平臺上都有java虛擬機(jvm),java編譯的中間文件class是由java虛擬機在運行時動態轉換為對應平臺的機器代碼 2.描述jvm的運行原理: 答: Java平臺由Java虛 ...
  • #QQ:[email protected]#本截圖適合安康碼截圖,如需其他地區截圖統計,可與我QQ或QQ郵箱聯繫#1、在當前文件夾下創建imgs文件夾用於存放圖片,圖片格式.jpg#2、在當前文件夾下創建“shuju.xlsx”的Excel用於存放統計結果文件夾目錄樣式 統計結果Excel樣式 具體代 ...
  • 昨天有個粉絲加了我,問我如何實現類似shiro的資源許可權表達式的訪問控制。我以前有一個小框架用的就是shiro,許可權控制就用了資源許可權表達式,所以這個東西對我不陌生,但是在Spring Security中我並沒有使用過它,不過我認為Spring Security可以實現這一點。是的,我找到了實現它的 ...
  • Python 能做很多無聊,但有意思的事情,例如接下來的一些案例。 Python 整蠱程式 以下程式,不要發代碼,要不實現不了你整蠱的目的。 要打包成一個 exe 程式,發給朋友才有意思。 使用 pip install pyinstaller。 打包命令如下: pyinstaller -F 文件名. ...
  • 引用 引用的基本使用 作用:給變數起別名 語法:數據類型 &別名 = 原名 引用註意事項 引用必須初始化 引用一旦初始化,就不可以更改(一個別名只能初始化一次:只可以跟一個原名) 引用做函數參數 函數傳參時,可以用引用技術讓形參修飾實參 簡化指針修飾形參 #include<iostream> usi ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...