日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

修改meta标签 查看源码没效果怎么办_Spring 源码学习(三)-自定义标签

發布時間:2025/3/21 javascript 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 修改meta标签 查看源码没效果怎么办_Spring 源码学习(三)-自定义标签 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

又來填坑啦,上一篇講完默認標簽的解析,這篇筆記記錄一下自定義標簽的解析吧。

我們知道,Spring 源碼的核心模塊是 Spring-core 和 Spring-beans,在此基礎上衍生出其他模塊,例如 context、 cache、 tx 等模塊,都是根據這兩個基礎模塊進行擴展的。

聰明如你,應該想到我們代碼中常用的緩存注解 @Cacheable、事務注解 @Transaction,還有阿里巴巴的 RPC 中間件 Dubbo,在配置文件中通過 或者 進行服務注冊和訂閱,這些都都屬于 Spring 的自定義標簽的實現,通過自定義標簽可以實現更加強大的功能!

作為一個有追求的程序員,當然不能滿足于框架自帶默認的標簽,為了擴展性和配置化要求,這時候就需要學習自定義標簽和使用自定義標簽~


Table of Contents generated with DocToc

  • 官方例子
  • 自定義標簽使用定義普通的 POJO 組件定義 XSD 描述文件定義組件解析器創建處理類的注冊器編寫 spring.hanlders 和 spring.schemas 文件使用 Demo配置文件測試代碼小結
  • 自定義標簽解析① 獲取標簽的命名空間② 根據命名空間找到對應的 NamespaceHandler③ 調用自定義的 NamespaceHandler 進行解析
  • 總結
  • 參考資料

又來填坑啦,上一篇講完默認標簽的解析,這篇筆記記錄一下自定義標簽的解析吧。

我們知道,Spring 源碼的核心模塊是 Spring-core 和 Spring-beans,在此基礎上衍生出其他模塊,例如 context、 cache、 tx 等模塊,都是根據這兩個基礎模塊進行擴展的。

聰明如你,應該想到我們代碼中常用的緩存注解 @Cacheable、事務注解 @Transaction,還有阿里巴巴的 RPC 中間件 Dubbo,在配置文件中通過 或者 進行服務注冊和訂閱,這些都都屬于 Spring 的自定義標簽的實現,通過自定義標簽可以實現更加強大的功能!

作為一個有追求的程序員,當然不能滿足于框架自帶默認的標簽,為了擴展性和配置化要求,這時候就需要學習自定義標簽和使用自定義標簽~


官方例子

先來看一張源碼圖片(紅框框圈著是重點喲)

剛才說了緩存和事務,那就拿這兩個舉例,還有一個標簽 (這個我也不太清楚,網上查的資料也不多,所以按照我的理解大家跟說下)

首先我們看到, 和 都是自定義標簽,左一是配置文件,進行 bean 的定義,頂部的 xmlns 是命名空間,表示標簽所屬的定義文件,像事務、緩存、MVC 的命名空間都是固定的。

而 myname 相當于萬金油,既可以定義為事務,又可以定義為緩存,只要我們在命名空間中進行相應的定義就能正確的識別。這個就是我們待會要使用到的自定義標簽,通過命名空間定位到我們想要的處理邏輯。

中間的是緩存定義的 xsd 文件,通過 定義元素, 區間內定義屬性列表, 定義單個屬性,詳細分析可以看下注釋~

右邊的是事務定義的 xsd 文件,大體內容的跟中間一樣,雖然元素名稱 有相同的,但是下面的屬性定義是有所區別的。

所以我們對自定義注解有個大概的了解,xsd 描述文件是個其中一個關鍵,在配置文件頂部的命名空間是標簽進行解析時,進行定位的配置,當然還有處理器,下面使用時進行介紹。

不知道理解的對不對,如果有誤的話請大佬們指出,我會進行修改的!


自定義標簽使用

Spring 提供了可擴展的 Schema 的支持,擴展 Spring 自定義標簽配置需要以下幾個步驟:

  • 創建一個需要擴展的組件
  • 定義一個 XSD 描述文件
  • 創建一個文件,實現 BeanDefinitionParse 接口,用來解析 XSD 文件中的定義和組件定義。
  • 創建一個 Handler 文件,擴展自 NamespaceHandlerSupport,將組件注冊到 Spring 容器
  • 編寫 Spring.handlers 和 Spring.schemas 文件

剛開始看到這些流程時,我還是有點慌的,畢竟從一個使用默認標簽的萌新小白,突然要我自己定義,感覺到很新鮮,所以請各位跟著下面的流程一起來看吧~


定義普通的 POJO 組件

這個沒啥好說的,就是一個普通的類:

public class Product {private Integer productId;private String unit;private String name;}

定義 XSD 描述文件

custom-product.xsd

我在上面的描述文件中,定義了一個新的 targetNamespace,同時定義了一個 叫 product 的新元素,并且將組件中的屬性都列在 中。XSD 文件是 XML DTD 的替代者,具體就不多深入,感興趣的同學可以繼續深入了解。


定義組件解析器

base.label.custom.ProductBeanDefinitionParser

public class ProductBeanDefinitionParser extends AbstractSingleBeanDefinitionParser {@Overrideprotected Class getBeanClass(Element element) {// 返回對應的類型return Product.class;}// 從 element 中解析并提取對應的元素@Overrideprotected void doParse(Element element, BeanDefinitionBuilder builder) {String productId = element.getAttribute("productId");String productName = element.getAttribute("name");String productUnit = element.getAttribute("unit");// 將提取到的數據放入 BeanDefinitionBuilder 中,等到完成所有 bean 的解析之后統一注冊到 beanFactory 中if (productId != null) {// element.getAttribute("") 方法取出來的都是 string 類型,使用時記得手動轉換builder.addPropertyValue("productId", Integer.valueOf(productId));}if (StringUtils.hasText(productName)) {builder.addPropertyValue("name", productName);}if (StringUtils.hasText(productUnit)) {builder.addPropertyValue("unit", productUnit);}}}

關鍵點在于,我們的解析器是繼承于 AbstractSingleBeanDefinitionParser,重載了兩個方法,詳細用途請看注釋~


創建處理類的注冊器

base.label.custom.ProductBeanHandler

public class ProductBeanHandler extends NamespaceHandlerSupport {@Overridepublic void init() {// 將組件解析器進行注冊到 `Spring` 容器registerBeanDefinitionParser("product", new ProductBeanDefinitionParser());}}

這個類也比較簡單,關鍵是繼承了 NamespaceHandlerSupport,對他進行了擴展,在該類初始化時將組件解析器進行注冊到 Spring 容器中。


編寫 spring.hanlders 和 spring.schemas 文件

我將文件位置放在 resources -> META-INF 目錄下:

spring.handlers

1http://vip-augus.github.io/schema/product=base.label.custom.ProductBeanHandler

spring.schemas

1http://vip-augus.github.io/schema/product.xsd=custom/custom-product.xsd

到了這一步,自定義的配置就結束了。下面是如何使用


使用 Demo

配置文件

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

測試代碼

public class ProductBootstrap {public static void main(String[] args) {ApplicationContext context = new ClassPathXmlApplicationContext("custom/custom-label.xml");Product product = (Product) context.getBean("product");// 輸出 Product{, productId ='1', unit='臺', name='Apple'}System.out.println(product.toString());}}

小結

現在來回顧一下,Spring 遇到自定義標簽是,加載自定義的大致流程:

  • 定位 spring.hanlders 和 spring.schemas:在兩個文件中找到對應的 handler 和 XSD,默認位置在 resources -> META-INF。
  • Handler 注冊 Parser:擴展了 NamespaceHandlerSupport 的類,在初始化注冊解析器
  • 運行解析器 Parser:擴展了 AbstractSingleBeanDefinitionParser,通過重載方法進行屬性解析,完成解析。

上面已經將自定義注解的使用講了,接下來講的是源碼中如何對自定義標簽進行解析。


自定義標簽解析

在上一篇筆記中,講了如何解析默認標簽,Spring 判斷一個標簽不是默認標簽的話,就會將這個標簽解析交給自定義標簽的解析方法

直接定位到解析自定義標簽的方法吧:

org.springframework.beans.factory.xml.BeanDefinitionParserDelegate#parseCustomElement(org.w3c.dom.Element, org.springframework.beans.factory.config.BeanDefinition)

public BeanDefinition parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) {// 注釋 3.8 ① 找到命名空間String namespaceUri = getNamespaceURI(ele);// ② 根據命名空間找到對應的 NamespaceHandlerNamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);// ③ 調用自定義的 NamespaceHandler 進行解析return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));}

看著流程是不是覺得很熟悉,我們剛才在自定義標簽使用時,定義的文件順序是一樣的,下面來講下這三個方法,具體代碼不會貼太多,主要記錄一些關鍵方法和流程,詳細代碼和流程請下載我上傳的工程~


① 獲取標簽的命名空間

public String getNamespaceURI(Node node) {return node.getNamespaceURI();}

這個方法具體做的事情很簡單,而且傳參的類型 org.w3c.dom.Node,已經提供了現成的方法,所以我們只需要調用即可。


② 根據命名空間找到對應的 NamespaceHandler

具體解析方法這這個類中:

org.springframework.beans.factory.xml.DefaultNamespaceHandlerResolver#resolve

public NamespaceHandler resolve(String namespaceUri) {// 注釋 3.9 獲取所有已經配置的 handler 映射Map handlerMappings = getHandlerMappings();// 從 map 中取出命名空間對應的 NamespaceHandler 的 className// 這個映射 map 值,沒有的話,會進行實例化類,然后放入 map,等下次同樣命名空間進來就能直接使用了Object handlerOrClassName = handlerMappings.get(namespaceUri);if (handlerOrClassName == null) {return null;}else if (handlerOrClassName instanceof NamespaceHandler) {return (NamespaceHandler) handlerOrClassName;}else {String className = (String) handlerOrClassName;Class> handlerClass = ClassUtils.forName(className, this.classLoader);if (!NamespaceHandler.class.isAssignableFrom(handlerClass)) {throw new FatalBeanException("Class [" + className + "] for namespace [" + namespaceUri +"] does not implement the [" + NamespaceHandler.class.getName() + "] interface");}// 實例化類NamespaceHandler namespaceHandler = (NamespaceHandler) BeanUtils.instantiateClass(handlerClass);// 調用 handler 的 init() 方法namespaceHandler.init();// 放入 handler 映射中handlerMappings.put(namespaceUri, namespaceHandler);return namespaceHandler;}}

找對應的 NamespaceHandler,關鍵方法在于 getHandlerMappings():

private Map getHandlerMappings() {Map handlerMappings = this.handlerMappings;// 如果沒有緩存,進行緩存加載,公共變量,加鎖進行操作,細節好評if (handlerMappings == null) {synchronized (this) {handlerMappings = this.handlerMappings;if (handlerMappings == null) {Properties mappings =PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);handlerMappings = new ConcurrentHashMap<>(mappings.size());CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);this.handlerMappings = handlerMappings;}}}return handlerMappings;}

所以我們能看到,找 Handler 時,使用的策略是延遲加載,在 map 緩存中找到了直接返回,沒找到對應的 Handler,將處理器實例化,執行 init() 方法,接著將 Handler 放入 map 緩存中,等待下一個使用。


③ 調用自定義的 NamespaceHandler 進行解析

回憶一下,我們在自定義標簽解析的時候,是沒有重載 parse() 方法,所以定位進去,看到實際調用方法是這兩行:

org.springframework.beans.factory.xml.NamespaceHandlerSupport#parse

public BeanDefinition parse(Element element, ParserContext parserContext) {// 尋找解析器并進行解析操作BeanDefinitionParser parser = findParserForElement(element, parserContext);// 真正解析調用調用的方法return (parser != null ? parser.parse(element, parserContext) : null);}

第一步獲取解析器,就是我們之前在 init() 方法中,注冊到 Spring 容器的解析器。

第二步才是解析器進行解析的方法,我們的解析器擴展的是 AbstractSingleBeanDefinitionParser,所以實際是調用了我們解析器父類的父類 AbstractBeanDefinitionParser 的 parse 方法:

org.springframework.beans.factory.xml.AbstractBeanDefinitionParser#parse

public final BeanDefinition parse(Element element, ParserContext parserContext) {// 注釋 3.10 實際自定義標簽解析器調用的方法,在 parseInternal 方法中,調用了我們重載的方法AbstractBeanDefinition definition = parseInternal(element, parserContext); ... return definition;}

解析關鍵方法

org.springframework.beans.factory.xml.AbstractSingleBeanDefinitionParser#parseInternal

protected final AbstractBeanDefinition parseInternal(Element element, ParserContext parserContext) {BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition();String parentName = getParentName(element);if (parentName != null) {builder.getRawBeanDefinition().setParentName(parentName);}Class> beanClass = getBeanClass(element);if (beanClass != null) {builder.getRawBeanDefinition().setBeanClass(beanClass);}else {String beanClassName = getBeanClassName(element);if (beanClassName != null) {builder.getRawBeanDefinition().setBeanClassName(beanClassName);}}builder.getRawBeanDefinition().setSource(parserContext.extractSource(element));BeanDefinition containingBd = parserContext.getContainingBeanDefinition();if (containingBd != null) {// Inner bean definition must receive same scope as containing bean.builder.setScope(containingBd.getScope());}if (parserContext.isDefaultLazyInit()) {// Default-lazy-init applies to custom bean definitions as well.builder.setLazyInit(true);}// 注釋 3.11 在這里調用了我們寫的解析方法doParse(element, parserContext, builder);return builder.getBeanDefinition();}

這里我要倒著講,在第二步解析時,不是直接調用了自定義的 doParse 方法,而是進行了一系列的數據準備,包括了 beanClass、 class、 lazyInit 等屬性的準備。

第一步解析,在我省略的代碼中,是將第二步解析后的結果進行包裝,從 AbstractBeanDefinition 轉換成 BeanDefinitionHolder ,然后進行注冊。轉換和注冊流程在第一篇筆記已經介紹過了,不再贅述。

到這里為止,我們自定義標簽的解析就完成了~

總結

以上是生活随笔為你收集整理的修改meta标签 查看源码没效果怎么办_Spring 源码学习(三)-自定义标签的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。