日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

如何使用Spring自定義Xml標簽

瀏覽:54日期:2023-07-08 14:25:59
目錄前言正文自定義NameSpaceHandler自定義schemaParserDecorator總結前言

在早期基于Xml配置的Spring Mvc項目中,我們往往會使用<context:component-scan basePackage=''>這種自定義標簽來掃描我們在basePackae配置里的包名下的類,并且會判斷這個類是否要注入到Spring容器中(比如這個類上標記了@Component注解就代表需要被Spring注入),如果需要那么它會幫助我們把這些類一一注入。

正文

在分析這個自定義標簽的解析機制前,我先提前劇透這個自定義標簽是通過哪個強大的類來解析的吧,就是隸屬于spring-context包下的ComponentScanBeanDefinitionParser,這個類在Springboot掃描Bean的過程中也扮演了重要角色。

既然知道了是這個類解析的,那么我們可以通過idea強大的搜索功能來搜它的引用之處了,這邊就截圖如下:

如何使用Spring自定義Xml標簽

可以看到這里面初始化了8個帶Parser后綴的各種Parser,從方法registerBeanDefinitionParser看出Spring是通過這個ContextNamespaceHandler來完成對以<context:自定義命名空間開頭的標簽解析器的注冊。我們可以看到Spring內部已經集成了幾個常用的NamespaceHandler,截圖如下:

如何使用Spring自定義Xml標簽

那么我們自己是否可以自定義一個NamespaceHandler來注冊我們自定義的標簽解析器呢?答案是肯定的。

自定義NameSpaceHandler

final class TestNamespaceHandler extends NamespaceHandlerSupport { @Override public void init() { //注冊parser registerBeanDefinitionParser('testBean', new TestBeanDefinitionParser()); registerBeanDefinitionParser('person', new PersonDefinitionParser()); //注冊element的 decorater registerBeanDefinitionDecorator('set', new PropertyModifyingBeanDefinitionDecorator()); registerBeanDefinitionDecorator('debug', new DebugBeanDefinitionDecorator()); //注冊 attr的 decorator registerBeanDefinitionDecoratorForAttribute('object-name', new ObjectNameBeanDefinitionDecorator()); }

到這里大家可能會有個疑問,這個NameSpaceHandler是怎么使用的呢?大家如果看了我之前寫的文章,那就會知道有一種方式可以配置我們自定義的NamespaceHandler.

public class CustomXmlApplicationContext extends AbstractXmlApplicationContext { private static final String CLASSNAME = CustomXmlApplicationContext.class.getSimpleName(); private static final String FQ_PATH = 'org/wonder/frame/customBean'; private static final String NS_PROPS = format('%s/%s.properties', FQ_PATH, CLASSNAME); public CustomXmlApplicationContext(String... configLocations) { setConfigLocations(configLocations); refresh(); } @Override protected void initBeanDefinitionReader(XmlBeanDefinitionReader reader) { super.initBeanDefinitionReader(reader); //1.指定resolver的 handlerMappingsLocation 就是 NamespaceHandler的 配置文件路徑 NamespaceHandlerResolver resolver = new DefaultNamespaceHandlerResolver(this.getClassLoader(), NS_PROPS); //2.設置resolver reader.setNamespaceHandlerResolver(resolver); //3.設置驗證模式 reader.setValidationMode(XmlBeanDefinitionReader.VALIDATION_XSD); //4.設置entityResolver reader.setEntityResolver(new CustomSchemaResolver()); }

可以看到我們在初始化BeanDefinitionReader的時候我們可以設置NamespaceHandlerResolver并且配置它的NamespaceHandler文件路徑。那這個NamespaceHandler配置文件應該怎么寫呢?

http://www.john.com/resource=org.wonder.frame.customBean.TestNamespaceHandler

就一行配置,但是這里有兩點要注意:

http://www.john.com/resource要和xsd的targetNamspace一致。 http://www.john.com/resource要和xml配置文件中的自定義namespace一致。

從代碼里看出來我們解析自定義標簽的時候其實是還需要自定義schema才能完成的。

自定義schema

private static final String CLASSNAME = CustomXmlApplicationContext.class.getSimpleName();private static final String FQ_PATH = 'org/wonder/frame/customBean';private static final String TEST_XSD = format('%s/%s.xsd', FQ_PATH, CLASSNAME);private final class CustomSchemaResolver extends PluggableSchemaResolver { public CustomSchemaResolver() { super(CustomXmlApplicationContext.this.getClassLoader()); } @Override public InputSource resolveEntity(String publicId, String systemId) throws IOException { InputSource source = super.resolveEntity(publicId, systemId); if (source == null) { try{ //todo 指定了xsd路徑 Resource resource = new ClassPathResource(TEST_XSD); source = new InputSource(resource.getInputStream()); source.setPublicId(publicId); source.setSystemId(systemId); return source; } catch (FileNotFoundException ex){ } } return null; }}

這里我們也通過ClassPathResource設置了自定義的xsd文件路徑。我們來看看xsd文件長啥樣:

<?xml version='1.0' encoding='UTF-8' standalone='no'?><xsd:schema xmlns='http://www.john.com/resource' xmlns:xsd='http://www.w3.org/2001/XMLSchema' targetNamespace='http://www.john.com/resource' elementFormDefault='qualified'> <xsd:element name='person'> <xsd:complexType> <xsd:attribute name='id' type='xsd:string' use='optional' form='unqualified'/> <xsd:attribute name='name' type='xsd:string' use='required' form='unqualified'/> <xsd:attribute name='age' type='xsd:integer' use='required' form='unqualified'/> </xsd:complexType> </xsd:element> <xsd:element name='testBean'> <xsd:complexType> <xsd:attribute name='id' type='xsd:string' use='required' form='unqualified'/> <xsd:attribute name='name' type='xsd:string' use='required' form='unqualified'/> <xsd:attribute name='age' type='xsd:integer' use='required' form='unqualified'/> </xsd:complexType> </xsd:element> <xsd:element name='set'> <xsd:complexType> <xsd:attribute name='name' type='xsd:string' use='required' form='unqualified'/> <xsd:attribute name='age' type='xsd:integer' use='required' form='unqualified'/> </xsd:complexType> </xsd:element> <xsd:element name='debug'/> <xsd:attribute name='object-name' type='xsd:string'/></xsd:schema>Parser

我們先來分析下TestBeanDefinitionParser和PersonDefinitionParser這兩者有啥區別:

TestBeanDefinitionParser直接實現了BeanDefinitionParser接口,內部直接定義一個RootBeanDefinition并且注冊。

private static class TestBeanDefinitionParser implements BeanDefinitionParser { @Override public BeanDefinition parse(Element element, ParserContext parserContext) { RootBeanDefinition definition = new RootBeanDefinition(); definition.setBeanClass(CustomBean.class); MutablePropertyValues mpvs = new MutablePropertyValues(); mpvs.add('name', element.getAttribute('name')); mpvs.add('age', element.getAttribute('age')); //1.設置beanDefinition的 屬性 propertyValues definition.setPropertyValues(mpvs); //2.獲取到beanDefinition的 registry parserContext.getRegistry().registerBeanDefinition(element.getAttribute('id'), definition); return null; }}

PersonDefinitionParser繼承自AbstractSingleBeanDefinitionParser抽象類,內部使用BeanDefinitionBuilder構造器來完成BeanDefinition的創建。

private static final class PersonDefinitionParser extends AbstractSingleBeanDefinitionParser { @Override protected Class<?> getBeanClass(Element element) { return CustomBean.class; } @Override protected void doParse(Element element, BeanDefinitionBuilder builder) { builder.addPropertyValue('name', element.getAttribute('name')); builder.addPropertyValue('age', element.getAttribute('age')); }}Decorator

我們看到在NameSpaceHandler中我們除了parser外還可以定義自定義元素的decorator和自定義attribute的decorator,那這兩個decorator是用來干嘛的呢?我們先來看下上述代碼中的PropertyModifyingBeanDefinitionDecorator。

private static class PropertyModifyingBeanDefinitionDecorator implements BeanDefinitionDecorator { @Override public BeanDefinitionHolder decorate(Node node, BeanDefinitionHolder definition, ParserContext parserContext) { Element element = (Element) node; //1.獲取BeanDefinition BeanDefinition def = definition.getBeanDefinition(); MutablePropertyValues mpvs = (def.getPropertyValues() == null) ? new MutablePropertyValues() : def.getPropertyValues(); mpvs.add('name', element.getAttribute('name')); mpvs.add('age', element.getAttribute('age')); ((AbstractBeanDefinition) def).setPropertyValues(mpvs); return definition; }}

從decorate方法內部看出這個decorator是用來給我們的BeanDefinition來添加屬性的。這樣一來我們就可以在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:test='http://www.john.com/resource' xmlns:util='http://www.springframework.org/schema/util' xsi:schemaLocation='http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-2.0.xsd http://www.john.com/resource http://www.john.com/resource/org/wonder/frame/customBean/CustomXmlApplicationContext.xsd' default-lazy-init='true'> <test:testBean name='Rob Harrop' age='23'/> <bean class='org.wonder.frame.customBean.CustomBean'> <!-- 自定義標簽加 自定義屬性 --> <test:set name='John wonder' age='36'/> </bean></beans>

我們看到testBean這個自定義標簽定義了兩個屬性name和age。之后我們在使用這個testBean的時候就可以獲取到它的name和age屬性了。

CustomBean bean = (CustomBean) beanFactory.getBean('testBean');System.out.println('name is:' +bean.getName() +' and age is:'+ bean.getAge());

那么ObjectNameBeanDefinitionDecorator這個attribute的Decorator是干嘛的呢?看如下示例

<!--為bean設置自定義Attr--><bean test:object-name='foo'/>

我們可以為這個Bean添加自定義Attribute,那么添加了這個Attribute我們怎么使用呢?看如下示例:

BeanDefinition beanDefinition = this.beanFactory.getBeanDefinition('decorateWithAttribute');assertEquals('foo', beanDefinition.getAttribute('objectName'));

我們通過BeanDefinition的getAttribute就能獲取到這個attribute值。

從Spring源碼得知BeanDefinition擴展了AttributeAccessor接口,這個接口是用于附加和訪問Bean元數據的通用的接口。直接實現這個接口的是AttributeAccessorSupport類。這個類里定義了名為attributes 的LinkedHashMap。

如何使用Spring自定義Xml標簽

總結

Spring通過自定義標簽和自定義屬性實現了很多擴展功能,很多我們常用的Spring配置內部都是通過它來完成的。

以上就是如何使用Spring自定義Xml標簽的詳細內容,更多關于使用Spring自定義Xml標簽的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
激情综合网五月| 国产亚洲精品美女久久| 99视频精品全部免费在线视频| 日韩精品免费一区二区三区| 黄色日韩在线| 91成人在线| 成人美女视频| 丝瓜av网站精品一区二区 | 久久国产三级| 久久精品国产亚洲一区二区三区| av高清一区| 亚洲精品在线观看91| 综合国产精品| 日韩福利视频导航| 98精品视频| 免费人成黄页网站在线一区二区 | 99国产精品私拍| 国产日韩一区二区三区在线| 日韩国产欧美一区二区| 国产精品免费看| 国产精品极品国产中出| 美女少妇全过程你懂的久久| 日韩在线成人| 亚洲人成在线网站| 午夜亚洲福利| 四虎成人av| 免费在线成人网| 国产资源在线观看入口av| 日韩在线卡一卡二| 国产精品久久观看| 玖玖精品视频| 国产精品毛片久久| 人人爽香蕉精品| 捆绑调教日本一区二区三区| 综合亚洲自拍| 日本欧美不卡| 国产私拍福利精品视频二区| 亚洲作爱视频| 久久久久久网| 国产亚洲一区二区三区不卡| 久久一区二区三区电影| 国产精品一区二区av日韩在线| 欧美日韩激情| 国产aⅴ精品一区二区三区久久| 亚洲日韩视频| 亚洲福利国产| 国产一区二区三区日韩精品| 婷婷久久免费视频| 亚洲天堂久久| 精品福利久久久| 91精品国产经典在线观看| 欧美日韩精品免费观看视频完整| 久久伊人久久| 日韩激情av在线| 在线一区视频| 97精品国产| 国产精品2023| 日本一不卡视频| 日韩一区二区免费看| 午夜欧美巨大性欧美巨大| 国产精品久久久久久久久久白浆| 视频在线在亚洲| 久久视频精品| 亚洲国产福利| 欧美久久一区二区三区| 综合国产视频| 国产成人77亚洲精品www| 久久国产人妖系列| 国产欧美精品| 国产精品1区在线| 国产精品一线天粉嫩av| 国产欧美一区二区三区国产幕精品| 日韩在线麻豆| 日韩av中文字幕一区二区 | 精品国产精品久久一区免费式 | 欧美专区一区二区三区| 鲁大师影院一区二区三区| 香蕉久久国产| 日本在线视频一区二区| 91伊人久久| 国产精品午夜一区二区三区| 麻豆一区二区三| 国产精品99一区二区三区| 精品中文字幕一区二区三区| 精品国产亚洲一区二区三区在线| 成人午夜网址| 久久人人88| 免费欧美日韩| 91麻豆精品激情在线观看最新| 国产精品亚洲人成在99www| 精品视频一区二区三区在线观看| 欧美另类中文字幕| 久久99高清| 日韩在线视频精品| 今天的高清视频免费播放成人| 老鸭窝亚洲一区二区三区| 欧美一级网站| 高清在线一区| 99久久99视频只有精品| 亚洲一区国产一区| 日本特黄久久久高潮| 精品久久97| 婷婷色综合网| 日本va欧美va瓶| 国产精品久久久网站| 国产精品一区二区精品| 国产极品嫩模在线观看91精品| 国产精品久久久久久久久久白浆 | 国产精品高颜值在线观看| 成人在线视频免费看| 日韩影院二区| 久久久9色精品国产一区二区三区| 私拍精品福利视频在线一区| 亚洲va在线| 亚洲一卡久久| 日韩精品一级| 欧美精品二区| 一区二区三区四区日本视频| 欧美+日本+国产+在线a∨观看| 国产视频一区欧美| 亚洲九九精品| 日韩影院精彩在线| 日韩高清成人在线| 欧美黑人做爰爽爽爽| 日韩精品电影| 亚洲视频国产| 国产 日韩 欧美 综合 一区| 蜜臀久久99精品久久一区二区| 日韩av一区二| 久久久久免费av| 91成人小视频| 九九久久婷婷| 国产精品嫩模av在线| 亚洲午夜久久久久久尤物| 97久久超碰| 亚洲女同中文字幕| 久久的色偷偷| 免费日韩av片| 国产欧洲在线| 日本在线不卡视频| 久久蜜桃资源一区二区老牛| 日韩激情网站| 成人在线网站| 国产精区一区二区| 伊人久久大香线蕉av超碰演员| 国产精品多人| 天堂成人免费av电影一区 | 亚洲小说欧美另类婷婷| 国产三级一区| 久久亚洲影院| 伊人久久在线| 欧美伊人久久| 国产精品丝袜xxxxxxx| 成人在线视频区| 日韩一区二区三区四区五区| 九九久久婷婷| 老牛影视精品| 国产激情欧美| 日韩精品欧美精品| 蜜桃视频欧美| 国产精品精品国产一区二区| 欧美日韩1区2区3区| 国产一区二区精品| 日韩欧美一区二区三区在线观看 | 日韩1区2区3区| 亚洲五月婷婷| 狠狠久久伊人| 91成人在线精品视频| 欧美中文日韩| 欧美日韩中文字幕一区二区三区 | 日韩精品看片| 激情中国色综合| 国产亚洲一区| 日韩欧美激情| 一区二区三区国产盗摄| 激情综合亚洲| 国产96在线亚洲| 亚洲成人国产| 国产在线欧美| 国产精品入口久久| 综合激情一区| 伊人精品一区| 国产一区二区三区四区五区| 日韩久久99| 午夜亚洲福利在线老司机| 欧美日韩国产观看视频| 欧美精品国产一区| 国产麻豆综合| 99久久久久国产精品| 欧美一区自拍| 丝袜美腿成人在线| 欧美99久久| 麻豆视频在线看| 青青伊人久久| 老司机精品久久| 免费观看不卡av| 亚洲成人不卡| 成人在线视频免费| 麻豆精品国产91久久久久久| 69精品国产久热在线观看|