亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

如何使用Spring自定義Xml標(biāo)簽

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

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

正文

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

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

如何使用Spring自定義Xml標(biāo)簽

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

如何使用Spring自定義Xml標(biāo)簽

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

自定義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()); }

到這里大家可能會有個(gè)疑問,這個(gè)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.設(shè)置resolver reader.setNamespaceHandlerResolver(resolver); //3.設(shè)置驗(yàn)證模式 reader.setValidationMode(XmlBeanDefinitionReader.VALIDATION_XSD); //4.設(shè)置entityResolver reader.setEntityResolver(new CustomSchemaResolver()); }

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

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

就一行配置,但是這里有兩點(diǎn)要注意:

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

從代碼里看出來我們解析自定義標(biāo)簽的時(shí)候其實(shí)是還需要自定義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設(shè)置了自定義的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這兩者有啥區(qū)別:

TestBeanDefinitionParser直接實(shí)現(xiàn)了BeanDefinitionParser接口,內(nèi)部直接定義一個(gè)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.設(shè)置beanDefinition的 屬性 propertyValues definition.setPropertyValues(mpvs); //2.獲取到beanDefinition的 registry parserContext.getRegistry().registerBeanDefinition(element.getAttribute('id'), definition); return null; }}

PersonDefinitionParser繼承自AbstractSingleBeanDefinitionParser抽象類,內(nèi)部使用BeanDefinitionBuilder構(gòu)造器來完成BeanDefinition的創(chuàng)建。

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,那這兩個(gè)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方法內(nèi)部看出這個(gè)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'> <!-- 自定義標(biāo)簽加 自定義屬性 --> <test:set name='John wonder' age='36'/> </bean></beans>

我們看到testBean這個(gè)自定義標(biāo)簽定義了兩個(gè)屬性name和age。之后我們在使用這個(gè)testBean的時(shí)候就可以獲取到它的name和age屬性了。

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

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

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

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

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

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

從Spring源碼得知BeanDefinition擴(kuò)展了AttributeAccessor接口,這個(gè)接口是用于附加和訪問Bean元數(shù)據(jù)的通用的接口。直接實(shí)現(xiàn)這個(gè)接口的是AttributeAccessorSupport類。這個(gè)類里定義了名為attributes 的LinkedHashMap。

如何使用Spring自定義Xml標(biāo)簽

總結(jié)

Spring通過自定義標(biāo)簽和自定義屬性實(shí)現(xiàn)了很多擴(kuò)展功能,很多我們常用的Spring配置內(nèi)部都是通過它來完成的。

以上就是如何使用Spring自定義Xml標(biāo)簽的詳細(xì)內(nèi)容,更多關(guān)于使用Spring自定義Xml標(biāo)簽的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 黄色一级毛片在线观看 | 天天干影视 | 高清色黄毛片一级毛片 | 美国毛片一级视频在线aa | 欧美a级片免费看 | 中文字幕第98页小明免费 | 91综合精品网站久久 | 色先锋av资源中文字幕 | 九九九精品在线观看 | 亚洲精品午夜aaa级久久久久 | 日本黄色一级网站 | 国产 另类 在线 欧美日韩 | 欧美亚洲尤物久久精品 | 成人午夜在线视频 | 亚洲精品二区 | 黄色美女毛片 | 一级毛片aa高清免费观看 | 在线观看精品一区 | 国产精品线在线精品国语 | 免费中文字幕在线国语 | 91天堂一区二区 | 99久久亚洲国产高清观看 | a级亚洲片精品久久久久久久 | 国产精品国产香蕉在线观看网 | 亚洲精品影院 | 亚洲欧美另类日本久久影院 | 国产成人ae在线观看网站站 | 亚洲欧美日产综合一区二区三区 | 青青操国产视频 | 国产三级网站在线观看 | 午夜成年人视频 | 香蕉免费一区二区三区在线观看 | 国产91青青成人a在线 | 国产乱理伦片在线观看 | 中文字幕不卡一区 二区三区 | 国产免费一级在线观看 | 国产三级毛片 | 不卡视频一区二区 | 毛片xxx | 欧美国产在线视频 | 国产色视频一区二区三区 |