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

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

詳解Spring如何解析占位符

瀏覽:135日期:2023-07-08 15:46:46
目錄什么是Spring的占位符?Spring什么時候去解析并占位符什么是Spring的占位符?

在以前的Spring 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:context='http://www.springframework.org/schema/context' xsi:schemaLocation='http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd> <context:property-placeholder ignore-unresolvable='true' location='classpath:jdbc.properties'/> <bean ><property name='url' value='${jdbc.url}'/> </bean></beans>

在上面的配置中jdbc這個Bean的url屬性值${jdbc.url}就代表占位符,占位符的真實值就存放在上述配置中的自定義元素的location屬性所代表的配置文件jdbc.properties中,這個配置文件里就一行內(nèi)容:

jdbc.url=127.0.0.1

那問題就來了,Spring又是在什么階段去解析并且把占位符替換為實際值的呢?

Spring什么時候去解析并占位符

從我們就可以知道它是一個自定義xml標(biāo)簽,那Spring勢必要解析它,我就直接黏貼Spring解析這個自定義元素的入口代碼給各位看官。該代碼就在BeanDefinitionParserDelegate這個類中:

/** * Parse the elements at the root level in the document: * 'import', 'alias', 'bean'. * @param root the DOM root element of the document */ //todo doRegisterBeanDefinitions -> parseBeanDefinitions -> parseDefaultElement protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {if (delegate.isDefaultNamespace(root)) { NodeList nl = root.getChildNodes(); for (int i = 0; i < nl.getLength(); i++) {Node node = nl.item(i);if (node instanceof Element) { Element ele = (Element) node; //如果屬于beans命名空間 if (delegate.isDefaultNamespace(ele)) {//處理默認(rèn)標(biāo)簽parseDefaultElement(ele, delegate); } else {//自定義標(biāo)簽//用到了parser//todo parser內(nèi)部 去注冊BeanDefinition 2021-3-15delegate.parseCustomElement(ele); }} }}else { delegate.parseCustomElement(root);} }

主要關(guān)注點:delegate.parseCustomElement(ele);

@Nullable public BeanDefinition parseCustomElement(Element ele) {return parseCustomElement(ele, null); } //todo property 子元素 也有可能 解析自定義元素 parsePropertySubElement @Nullable public BeanDefinition parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) {String namespaceUri = getNamespaceURI(ele);if (namespaceUri == null) { return null;}//resolve里有初始化過程//根據(jù)命名空間uri獲取 NamespaceHandler//todo 獲取命名空間下的自定義handler 比如 ContextNamespaceHandlerNamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);if (handler == null) { //todo 如果在spring.handlers配置文件 里沒有定義這個命令空間的handler就會 報這個錯 2020-09-14 error('Unable to locate Spring NamespaceHandler for XML schema namespace [' + namespaceUri + ']', ele); return null;}//調(diào)用parse方法//這里ParserContext注入registry//readerContext里 reader->XmlBeanDefinitionReader 里包含了 registry//TODO 會初始化一個ParserContext進去return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd)); }

到這里我們又要關(guān)注這個NamespaceHandler是怎么獲取到的,從上面代碼可知,Spring會從當(dāng)前readerContext獲取到NamespaceHandlerResolver后通過其resolve方法并根據(jù)傳入的當(dāng)前namespaceUri就可以獲得當(dāng)前適合的NamespaceHandler。

/** * Create the {@link XmlReaderContext} to pass over to the document reader. */ public XmlReaderContext createReaderContext(Resource resource) {//把當(dāng)前reader放進去 ,reader存放了 beanRegistry//beanRegistry 里定義了 registerBeanDefinition 這個重要的方法return new XmlReaderContext(resource, this.problemReporter, this.eventListener,this.sourceExtractor, this, getNamespaceHandlerResolver()); } /** * Lazily create a default NamespaceHandlerResolver, if not set before. * 解析自定義標(biāo)簽時用到 * @see #createDefaultNamespaceHandlerResolver() */ public NamespaceHandlerResolver getNamespaceHandlerResolver() {if (this.namespaceHandlerResolver == null) { this.namespaceHandlerResolver = createDefaultNamespaceHandlerResolver();}return this.namespaceHandlerResolver; } /** * Create the default implementation of {@link NamespaceHandlerResolver} used if none is specified. * <p>The default implementation returns an instance of {@link DefaultNamespaceHandlerResolver}. * @see DefaultNamespaceHandlerResolver#DefaultNamespaceHandlerResolver(ClassLoader) */ protected NamespaceHandlerResolver createDefaultNamespaceHandlerResolver() {ClassLoader cl = (getResourceLoader() != null ? getResourceLoader().getClassLoader() : getBeanClassLoader());return new DefaultNamespaceHandlerResolver(cl); } //返回默認(rèn)的 public DefaultNamespaceHandlerResolver(@Nullable ClassLoader classLoader) {this(classLoader, DEFAULT_HANDLER_MAPPINGS_LOCATION); }

從上面的代碼中可知Spring使用的是默認(rèn)的DefaultNamespaceHandlerResolver,它當(dāng)然也給開發(fā)者留了自定義NamespaceHandlerResolver的機會。那我們現(xiàn)在就可以看看DefaultNamespaceHandlerResolver如何根據(jù)namespaceUri解析到對應(yīng)的NamespaceHandler的。

首先獲取到的就是context命名空間,完整路徑為http://www.springframework.org/schema/context。我們從DefaultNamespaceHandlerResolver類中可以看到它是如何解析這個命名空間的。

/** * Locate the {@link NamespaceHandler} for the supplied namespace URI * from the configured mappings. * @param namespaceUri the relevant namespace URI * @return the located {@link NamespaceHandler}, or {@code null} if none found */ @Override @Nullable public NamespaceHandler resolve(String namespaceUri) {Map<String, Object> handlerMappings = getHandlerMappings();Object handlerOrClassName = handlerMappings.get(namespaceUri);if (handlerOrClassName == null) { return null;}else if (handlerOrClassName instanceof NamespaceHandler) { return (NamespaceHandler) handlerOrClassName;}else { String className = (String) handlerOrClassName; try {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);//todo 命名空間處理器 調(diào)用初始化過程 2020-09-04namespaceHandler.init();handlerMappings.put(namespaceUri, namespaceHandler);return namespaceHandler; } catch (ClassNotFoundException ex) {throw new FatalBeanException('Could not find NamespaceHandler class [' + className +'] for namespace [' + namespaceUri + ']', ex); } catch (LinkageError err) {throw new FatalBeanException('Unresolvable class definition for NamespaceHandler class [' +className + '] for namespace [' + namespaceUri + ']', err); }} } /** * Load the specified NamespaceHandler mappings lazily. */ private Map<String, Object> getHandlerMappings() {Map<String, Object> handlerMappings = this.handlerMappings;if (handlerMappings == null) { synchronized (this) {handlerMappings = this.handlerMappings;if (handlerMappings == null) { if (logger.isTraceEnabled()) {logger.trace('Loading NamespaceHandler mappings from [' + this.handlerMappingsLocation + ']'); } try {//todo handlerMappings為空 才去 獲取所有屬性映射 2020-09-04//spring-aop spring-beans spring-contextProperties mappings =PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);if (logger.isTraceEnabled()) { logger.trace('Loaded NamespaceHandler mappings: ' + mappings);}handlerMappings = new ConcurrentHashMap<>(mappings.size());CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);this.handlerMappings = handlerMappings; } catch (IOException ex) {throw new IllegalStateException('Unable to load NamespaceHandler mappings from location [' + this.handlerMappingsLocation + ']', ex); }} }}return handlerMappings; }

上面代碼中的handlerMappingsLocation一般就是Spring默認(rèn)的路徑:

//指定了默認(rèn)的handler路徑 ,可以傳入指定路徑改變 /** * The location to look for the mapping files. Can be present in multiple JAR files. */ public static final String DEFAULT_HANDLER_MAPPINGS_LOCATION = 'META-INF/spring.handlers';

我們就回到spring-context項目工程下的resoures/META-INF文件夾下的spring.handlers文件里定義了如下規(guī)則:

http://www.springframework.org/schema/context=org.springframework.context.config.ContextNamespaceHandlerhttp://www.springframework.org/schema/jee=org.springframework.ejb.config.JeeNamespaceHandlerhttp://www.springframework.org/schema/lang=org.springframework.scripting.config.LangNamespaceHandlerhttp://www.springframework.org/schema/task=org.springframework.scheduling.config.TaskNamespaceHandlerhttp://www.springframework.org/schema/cache=org.springframework.cache.config.CacheNamespaceHandler

可以看到context自定義命名空間就是對應(yīng)的ContextNamespaceHandler。我們打開這個類瞧一瞧:

public class ContextNamespaceHandler extends NamespaceHandlerSupport { @Override public void init() {//調(diào)用抽象類NamespaceHandlerSupport的注冊解析器方法registerBeanDefinitionParser('property-placeholder', new PropertyPlaceholderBeanDefinitionParser());registerBeanDefinitionParser('property-override', new PropertyOverrideBeanDefinitionParser());registerBeanDefinitionParser('annotation-config', new AnnotationConfigBeanDefinitionParser());registerBeanDefinitionParser('component-scan', new ComponentScanBeanDefinitionParser());registerBeanDefinitionParser('load-time-weaver', new LoadTimeWeaverBeanDefinitionParser());registerBeanDefinitionParser('spring-configured', new SpringConfiguredBeanDefinitionParser());registerBeanDefinitionParser('mbean-export', new MBeanExportBeanDefinitionParser());registerBeanDefinitionParser('mbean-server', new MBeanServerBeanDefinitionParser()); }}

發(fā)現(xiàn)它只定義了一個init方法,顧名思義就是初始化的意思,那想當(dāng)然的就是它啥時候會執(zhí)行初始化呢?我們回到DefaultNamespaceHandler的resolve方法,發(fā)現(xiàn)它內(nèi)部有一處namespaceHandler.init();, 這里就執(zhí)行了對應(yīng)命名空間處理器的初始化方法。接下來我們又要看看初始化了做了些啥,我們發(fā)現(xiàn)它調(diào)用了同一個方法registerBeanDefinitionParser也就是注冊

Bean定義解析器,到這里我們先按下暫停鍵,再捋下上面的整體流程:

Spring在解析自定義標(biāo)簽的時候會根據(jù)自定義命名空間去查找合適的NamespaceHandler. 自定義的NamespaceHandler是由NamespaceHandlerResolver去解析得到的。 NamespaceHandlerResolver會根據(jù)classLoader.getResources查找所有類路徑下的spring.handlers。 查找到后把文件內(nèi)容轉(zhuǎn)換成handlerMappings,然后根據(jù)傳入的自定義命名空間匹配到NamespaceHandler 執(zhí)行NamespaceHandler的init方法注冊BeanDefinitionParser。

那這個parser做了點什么強大的功能呢?我們下回分解。

以上就是詳解Spring如何解析占位符的詳細(xì)內(nèi)容,更多關(guān)于Spring 解析占位符的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
三级久久三级久久久| 亚洲一级少妇| 免费久久精品视频| 免费日韩av片| 日本欧美在线看| 另类av一区二区| 蜜臀久久99精品久久久久久9| 久久在线电影| 91久久久精品国产| 红桃视频亚洲| 蜜臀91精品一区二区三区| 999视频精品| 欧美日中文字幕| 国产模特精品视频久久久久| 天堂va蜜桃一区二区三区| 99xxxx成人网| 蜜桃一区二区三区在线| 男人的天堂亚洲一区| 亚洲欧洲av| 免费成人在线影院| 亚洲涩涩av| 国产日产精品_国产精品毛片| 国产日韩欧美在线播放不卡| 免费在线欧美黄色| 黑森林国产精品av| 99久精品视频在线观看视频| 精品久久99| 欧美日韩亚洲三区| 国产成人精品免费视| 日本一区福利在线| 久久精品理论片| 日韩一区二区在线免费| 亚洲精品乱码久久久久久蜜桃麻豆| 国产在线|日韩| 91精品福利观看| 欧美xxxx性| 热三久草你在线| 九色porny丨国产首页在线| 亚洲欧洲美洲av| 蜜臀国产一区二区三区在线播放 | 超碰99在线| 欧美日一区二区三区在线观看国产免 | 黄色欧美在线| 国产精品a久久久久| 首页欧美精品中文字幕| 日韩精品五月天| 国产精品av久久久久久麻豆网| 国产国产精品| 亚洲啊v在线免费视频| 日本aⅴ亚洲精品中文乱码| 国产调教精品| 欧美日韩国产一区精品一区| 国产美女高潮在线| 国产一区日韩| 久久久91麻豆精品国产一区| 日韩区欧美区| 久久午夜影视| 性一交一乱一区二区洋洋av| 99在线|亚洲一区二区| 亚洲二区免费| 激情久久久久久| 999久久久国产精品| 国产综合视频| 亚洲欧美久久| 在线观看精品| 日韩高清一区| 日本午夜精品视频在线观看| 天堂va在线高清一区| 国产精品毛片在线| 综合亚洲自拍| 水野朝阳av一区二区三区| 亚洲一区观看| 日韩成人精品一区二区三区| 一区二区91| 日韩毛片网站| 久久国产三级精品| 久久久夜夜夜| 日韩精彩视频在线观看| 黄在线观看免费网站ktv| 亚洲激情国产| 日本欧美韩国一区三区| 在线视频精品| 国产一区91| 久久久天天操| 久久国内精品| 亚洲午夜免费| 亚洲成人不卡| 国产精品综合色区在线观看| 国产视频久久| 日韩大片免费观看| 日本v片在线高清不卡在线观看| 日韩高清不卡| 精品中文字幕一区二区三区| 日韩精品一二三| 一区二区日韩免费看| 国内不卡的一区二区三区中文字幕| 久久97视频| 天使萌一区二区三区免费观看| 水蜜桃久久夜色精品一区的特点| 国产精品毛片一区二区在线看| 日韩久久精品| 99riav国产精品| 日韩深夜视频| 欧美日韩18| 亚洲精品**中文毛片| 福利在线一区| 蜜臀久久久久久久| 亚洲欧美日韩国产一区二区| 麻豆精品视频在线观看免费| 成人午夜精品| 久久亚洲视频| 日韩有码av| 亚洲经典在线| 日韩一区二区三区精品| 国产精品22p| 亚洲精品乱码久久久久久蜜桃麻豆| 欧美视频久久| 蜜桃久久av| 国产精品地址| 亚洲一区网站| 久久久精品网| 日本黄色精品| 国产精品成人**免费视频 | 亚洲人妖在线| 久久精品国内一区二区三区| 日韩精品久久久久久| 精品中文字幕一区二区三区四区| 国产精品99久久免费观看| 国产一区久久| 国产精品a级| 久久要要av| 国产日产精品_国产精品毛片 | 欧美日韩国产综合网| 日韩另类视频| 麻豆视频在线观看免费网站黄| 日韩成人综合| 99久久久久国产精品| 久久精品一区二区不卡| 亚洲先锋成人| 久久成人一区| 亚洲97av| 国产精品一区二区精品| 国产极品一区| 免费看久久久| 日韩激情一区| 自由日本语亚洲人高潮| 美女尤物久久精品| 91九色综合| 国产精品极品| 在线看片国产福利你懂的| 色婷婷精品视频| 日韩亚洲在线| 国产日韩欧美| av高清不卡| 水蜜桃久久夜色精品一区的特点| 日韩一区二区三免费高清在线观看| 国产欧美日韩免费观看| 视频在线不卡免费观看| 久久久777| 中文字幕日韩欧美精品高清在线| 久久国产精品免费精品3p| 国产成人精选| 9国产精品视频| 国产精品调教| 日韩精品免费一区二区三区| 国产精品毛片在线看| 日韩高清不卡在线| 国产一区二区三区四区五区| 久久人人99| 欧美在线精品一区| 午夜天堂精品久久久久| 狠狠久久伊人中文字幕| jiujiure精品视频播放| 日韩有吗在线观看| 国产一区二区三区日韩精品 | 日韩视频在线一区二区三区| 婷婷精品在线| 成人在线黄色| 亚洲一区二区动漫| 欧美激情三区| 日韩一级网站| 精品国产亚洲一区二区三区大结局| 99国产精品一区二区| 日韩av一区二| 久久精品免费一区二区三区 | 日韩精品一区二区三区免费视频| 精品久久久网| 亚洲在线成人| 高潮久久久久久久久久久久久久| 免费在线观看精品| 日本激情一区| 日韩欧美四区| 久久精品主播| 国产精品xxx在线观看| 亚洲五月婷婷| 国产aa精品| 好吊一区二区三区| 麻豆国产欧美一区二区三区| 欧美日韩视频| 97人人精品|