成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

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

spring是如何解析xml配置文件中的占位符

瀏覽:2日期:2023-08-02 11:16:20

前言

我們在配置Spring Xml配置文件的時候,可以在文件路徑字符串中加入 ${} 占位符,Spring會自動幫我們解析占位符,這么神奇的操作Spring是怎么幫我們完成的呢?這篇文章我們就來一步步揭秘。

1.示例

ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext();applicationContext.setConfigLocation('${java.version}.xml');applicationContext.refresh();String[] beanNames = applicationContext.getBeanDefinitionNames();for (String beanName : beanNames) { System.out.println(beanName);}

這段代碼在我工程里是會報錯的,如下:

Caused by: java.io.FileNotFoundException: class path resource [1.8.0_144.xml] cannot be opened because it does not exist at org.springframework.core.io.ClassPathResource.getInputStream(ClassPathResource.java:190) at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions(XmlBeanDefinitionReader.java:336) ... 11 more

可以看到報錯里面的文件路徑變成了1.8.0_144.xml,也就是說Spring幫我們把${java.version}解析成了實際值。

2.原理

AbstractRefreshableConfigApplicationContext我們在之前的文章里提到過這個類的resolve方法,我們再來瞧一眼:

/** * Resolve the given path, replacing placeholders with corresponding * environment property values if necessary. Applied to config locations. * @param path the original file path * @return the resolved file path * @see org.springframework.core.env.Environment#resolveRequiredPlaceholders(String) */ protected String resolvePath(String path) { //通過當前環境去 解析 必要的占位符 return getEnvironment().resolveRequiredPlaceholders(path); }

獲取當前環境,這個環境在示例代碼中就是 StandardEnvironment ,并且根據當前環境去解析占位符,這個占位符解析不到還會報錯。

resolveRequiredPlaceHolders由StandardEnvironment的父類AbstractEnvironment實現。

AbstractEnvironment

//把propertySources放入 Resolver中private final ConfigurablePropertyResolver propertyResolver = new PropertySourcesPropertyResolver(this.propertySources);@Overridepublic String resolveRequiredPlaceholders(String text) throws IllegalArgumentException { return this.propertyResolver.resolveRequiredPlaceholders(text);}

這里的propertySources很重要了,從命名也可以看出我們解析占位符的來源就是從這個集合中來的。這個集合是在我們StandardEnvironment實例化的時候去自定義的。

StandardEnvironment

/** * Create a new {@code Environment} instance, calling back to * {@link #customizePropertySources(MutablePropertySources)} during construction to * allow subclasses to contribute or manipulate(操作) {@link PropertySource} instances as * appropriate. * @see #customizePropertySources(MutablePropertySources) */ //StandardEnvironment 實例化調用 public AbstractEnvironment() { customizePropertySources(this.propertySources); }@Overrideprotected void customizePropertySources(MutablePropertySources propertySources) { //todo Java提供了System類的靜態方法getenv()和getProperty()用于返回系統相關的變量與屬性, //todo getenv方法返回的變量大多于系統相關, //todo getProperty方法返回的變量大多與java程序有關。 //https://www.cnblogs.com/Baronboy/p/6030443.html propertySources.addLast(new MapPropertySource(SYSTEM_PROPERTIES_PROPERTY_SOURCE_NAME, getSystemProperties())); //SystemEnvironmentPropertySource 是System.getenv() propertySources.addLast(new SystemEnvironmentPropertySource(SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, getSystemEnvironment()));}

最重要的肯定是我們的 propertyResolver.resolveRequiredPlaceholders 方法了,propertyResolver.resolveRequiredPlaceholders其實是PropertySourcesPropertyResolver的父類AbstractPropertyResolver來實現。

AbstractPropertyResolver

//創建一個占位符的helper去解析@Overridepublic String resolveRequiredPlaceholders(String text) throws IllegalArgumentException { if (this.strictHelper == null) { //不忽略 this.strictHelper = createPlaceholderHelper(false); } return doResolvePlaceholders(text, this.strictHelper);} //私有方法 //是否忽略 無法解決的占位符private PropertyPlaceholderHelper createPlaceholderHelper(boolean ignoreUnresolvablePlaceholders) { //默認使用${ placeholderPrefix return new PropertyPlaceholderHelper(this.placeholderPrefix, this.placeholderSuffix, this.valueSeparator, ignoreUnresolvablePlaceholders);}private String doResolvePlaceholders(String text, PropertyPlaceholderHelper helper) { //PlaceholderResolver function interface //todo important 重要的是這個getPropertyAsRawString return helper.replacePlaceholders(text, this::getPropertyAsRawString); }

這里的 this::getPropertyAsRawString 很重要,利用了java8的函數式接口來實現。它的定義在AbstractPropertyResolver里

/** * Retrieve the specified property as a raw String, * i.e. without resolution of nested placeholders. * @param key the property name to resolve * @return the property value or {@code null} if none found */ @Nullable protected abstract String getPropertyAsRawString(String key);

但是我們在doResolvePlaceholders里指向的this,所以還得看PropertySourcesPropertyResolver類。

PropertySourcesPropertyResolver

//提供給函數接口 PlaceholderResolver //todo 解析 xml配置文件路徑占位符的時候調用的是這個 2020-09-11 @Override @Nullable protected String getPropertyAsRawString(String key) { return getProperty(key, String.class, false); }@Nullable protected <T> T getProperty(String key, Class<T> targetValueType, boolean resolveNestedPlaceholders) { if (this.propertySources != null) { //例如遍歷的是MutablePropertySources 的propertySourceList for (PropertySource<?> propertySource : this.propertySources) { if (logger.isTraceEnabled()) { logger.trace('Searching for key ’' + key + '’ in PropertySource ’' + propertySource.getName() + '’'); } Object value = propertySource.getProperty(key); if (value != null) { //todo 解析 profile變量的時候 會去 解析 變量中的占位符 2020-09-11 //TODO 解析xml配置文件路徑字符串的時候 如果占位符 變量 的值 包含占位符 在這里 不會去解析 通過Helper 去解析 PropertyPlaceholderHelper if (resolveNestedPlaceholders && value instanceof String) { value = resolveNestedPlaceholders((String) value); } logKeyFound(key, propertySource, value); //跳出for 循環 return convertValueIfNecessary(value, targetValueType); } } } if (logger.isTraceEnabled()) { logger.trace('Could not find key ’' + key + '’ in any property source'); } return null; }

看到沒有,我們是遍歷this.propertySources集合,然后根據key調用它的getProperty方法獲取value。我們從上面的StandardEnvrionment中看到我們定義的是 MapPropertySource 和 SystemEnvironmentPropertySource .

MapPropertySource

//從source中取得屬性@Override@Nullablepublic Object getProperty(String name) { return this.source.get(name);}

這里的source就是getSystemProperties(),也就是 AbstractEnvironment中的方法:

@Override@SuppressWarnings({'unchecked', 'rawtypes'})public Map<String, Object> getSystemProperties() { try { //Hashtable return (Map) System.getProperties(); } catch (AccessControlException ex) { return (Map) new ReadOnlySystemAttributesMap() { @Override @Nullable protected String getSystemAttribute(String attributeName) { try { return System.getProperty(attributeName); } catch (AccessControlException ex) { if (logger.isInfoEnabled()) { logger.info('Caught AccessControlException when accessing system property ’' + attributeName + '’; its value will be returned [null]. Reason: ' + ex.getMessage()); } return null; } } }; }}

我們還忘了很重要的一步,就是PropertyPlaceholderHelper的replacePlaceholders方法。

PropertyPlaceholderHelper

//protected 范圍protected String parseStringValue( String value, PlaceholderResolver placeholderResolver, Set<String> visitedPlaceholders) { StringBuilder result = new StringBuilder(value); //如果value中沒有占位符前綴 那直接返回result int startIndex = value.indexOf(this.placeholderPrefix); while (startIndex != -1) { //找到占位符的最后一個索引 int endIndex = findPlaceholderEndIndex(result, startIndex); if (endIndex != -1) { String placeholder = result.substring(startIndex + this.placeholderPrefix.length(), endIndex); String originalPlaceholder = placeholder; if (!visitedPlaceholders.add(originalPlaceholder)) { throw new IllegalArgumentException( 'Circular placeholder reference ’' + originalPlaceholder + '’ in property definitions'); } //1. todo 2020-09-01 解析出來占位符,比如java.version //解析內嵌占位符 // Recursive invocation, parsing placeholders contained in the placeholder key. placeholder = parseStringValue(placeholder, placeholderResolver, visitedPlaceholders); // Now obtain the value for the fully resolved key... //2.todo 2020-09-01 獲取實際值 String propVal = placeholderResolver.resolvePlaceholder(placeholder); if (propVal == null && this.valueSeparator != null) { int separatorIndex = placeholder.indexOf(this.valueSeparator); if (separatorIndex != -1) { String actualPlaceholder = placeholder.substring(0, separatorIndex); String defaultValue = placeholder.substring(separatorIndex + this.valueSeparator.length()); //這里就是實際獲取占位符中值得地方。 propVal = placeholderResolver.resolvePlaceholder(actualPlaceholder); } } if (propVal != null) { //從占位符里獲取的值也有可能包含占位符 這里可能會報 Circular placeholder reference propVal = parseStringValue(propVal, placeholderResolver, visitedPlaceholders); //替換占位符 為 實際值 result.replace(startIndex, endIndex + this.placeholderSuffix.length(), propVal); if (logger.isTraceEnabled()) { logger.trace('Resolved placeholder ’' + placeholder + '’'); } startIndex = result.indexOf(this.placeholderPrefix, startIndex + propVal.length()); } //省略部分代碼 } else { startIndex = -1; } } return result.toString();}

到這里我們就可以看到Spring在處理一個小小的占位符就做了這么多設計。可見這個架構是如此嚴謹。下篇文章我們就來探討下Spring是如何加載這個Xml文件的。

以上就是spring是如何解析xml配置文件中的占位符的詳細內容,更多關于spring解析xml 占位符的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
91麻豆免费观看| 欧美日韩在线播放三区四区| 色婷婷综合久久久久中文一区二区| 国产精品久久国产精麻豆99网站| 不卡免费追剧大全电视剧网站| 欧美日韩国产高清一区| 亚洲成人av在线电影| 日韩视频免费| 亚洲免费av高清| 亚洲高清在线| 中文字幕在线观看一区二区| 国产综合第一页| 国产精品久久三区| 国产精品sss| 日本一区二区电影| 欧美日韩精品久久| 国产欧美日韩中文久久| 91天堂素人约啪| 国产人伦精品一区二区| 91在线视频免费91| 国产欧美一区二区三区沐欲| 欧美黄色一级视频| 国产欧美日韩精品a在线观看| www.日本不卡| 久久亚洲二区三区| 欧美99久久| 国产精品成人一区二区三区夜夜夜| 国产一区二区三区自拍| 亚洲日本韩国一区| 亚洲一区二区三区精品在线观看| 亚洲第一av色| 欧美性猛交一区二区三区精品| 七七婷婷婷婷精品国产| 欧美性色黄大片| 精品伊人久久久久7777人| 精品视频在线免费观看| 美国十次了思思久久精品导航| 欧美日韩视频不卡| 国产91精品精华液一区二区三区 | 中文精品在线| 三级在线观看一区二区| 欧美午夜一区二区| 成人免费不卡视频| 日本一区二区三区在线不卡| 尹人成人综合网| 亚洲电影你懂得| 欧美日韩高清影院| 不卡影院免费观看| 国产精品久久三区| 亚洲一区二区网站| 久久99精品久久只有精品| 日韩一区二区在线观看视频| 91蜜桃婷婷狠狠久久综合9色| 中文字幕中文字幕在线一区| 国产精品视频福利| 久久成人18免费观看| 日韩精品一区二区三区在线播放| 色综合天天综合网天天狠天天| 国产精品久久久久久久久图文区| 亚洲一区成人| 国产一区二区三区久久久| 久久亚洲一级片| 亚洲精品一品区二品区三品区| 亚洲va中文字幕| 欧美日韩久久不卡| 不卡的av中国片| 亚洲人成网站色在线观看| 欧美影视一区二区三区| 91欧美一区二区| 亚洲成人免费影院| 日韩亚洲欧美一区二区三区| 亚洲性感美女99在线| 日韩av在线免费观看不卡| 日韩欧美一级在线播放| 在线视频国内自拍亚洲视频| 麻豆成人91精品二区三区| 久久久久久亚洲综合影院红桃 | 欧美巨大另类极品videosbest | 亚洲久久成人| 国产精品影视在线观看| 国产精品视频一二三| 久久精品综合一区| 成人av在线一区二区三区| 亚洲卡通欧美制服中文| 欧美电影影音先锋| 国语自产精品视频在线看抢先版结局 | 一区二区欧美国产| 欧美丰满少妇xxxxx高潮对白 | 1区2区3区国产精品| 欧美网站大全在线观看| 欧美日本在线| 久久精品999| 中文字幕亚洲精品在线观看| 欧美日韩视频一区二区| 精品999网站| 国产在线视视频有精品| 亚洲免费在线电影| 欧美一级国产精品| 亚洲欧美日韩国产综合精品二区 | 一本色道久久综合亚洲精品高清| 国产呦萝稀缺另类资源| 国产精品白丝在线| 欧美日韩1区2区| 夜夜夜久久久| 99精品国产一区二区三区不卡| 亚洲成av人综合在线观看| 日韩精品一区二区三区老鸭窝| 国产一区二区三区久久久久久久久 | 国产精品资源在线观看| 亚洲在线免费播放| 欧美sm美女调教| 一本色道久久加勒比精品| 91伊人久久大香线蕉| 无吗不卡中文字幕| 国产日韩精品一区二区浪潮av| 91久久人澡人人添人人爽欧美| 欧美三区美女| 国产不卡视频在线观看| 天堂久久久久va久久久久| 国产亲近乱来精品视频| 欧美巨大另类极品videosbest| 国产精品一区二区你懂得| 91丝袜美腿高跟国产极品老师| 国产综合成人久久大片91| 亚洲444eee在线观看| 国产精品沙发午睡系列990531| 欧美一区二区精美| 色屁屁一区二区| 一区二区三区国产在线| 午夜国产一区| 成人免费电影视频| 国模冰冰炮一区二区| 婷婷六月综合亚洲| 中文字幕一区二区日韩精品绯色| 日韩三级.com| 欧美吻胸吃奶大尺度电影| 亚洲一区三区电影在线观看| 欧美午夜影院| 99久久国产综合精品色伊| 国内一区二区在线| 日韩精品欧美成人高清一区二区| 亚洲欧美日韩国产中文在线| 国产欧美日韩亚州综合 | 7777精品伊人久久久大香线蕉超级流畅| 亚洲女同同性videoxma| 亚洲国产一区二区在线| 欧美日韩国产在线一区| 91婷婷韩国欧美一区二区| 国产91精品入口| 精品一区二区三区免费观看| 天天色图综合网| 亚洲综合av网| 亚洲欧美自拍偷拍| 日本一区二区三区dvd视频在线| 欧美成人a∨高清免费观看| 欧美日韩国产综合久久| 色拍拍在线精品视频8848| 免费不卡亚洲欧美| 99视频+国产日韩欧美| 尤妮丝一区二区裸体视频| 91蝌蚪porny成人天涯| 不卡一区二区中文字幕| 国产成人午夜99999| 国产精品白丝jk白祙喷水网站| 黑人巨大精品欧美黑白配亚洲| 免费亚洲电影在线| 蜜臀久久久99精品久久久久久| 午夜精品久久久久久久蜜桃app| 亚洲综合成人网| 亚洲一区影音先锋| 亚洲一级在线观看| 一区二区三区不卡在线观看| 亚洲欧美综合网| 亚洲精品伦理在线| 尤物在线观看一区| 亚洲综合丝袜美腿| 一区二区三区高清| 亚洲电影激情视频网站| 亚洲一二三四区不卡| 五月天激情综合| 日韩国产欧美在线视频| 蜜桃av一区二区| 狠狠色丁香久久婷婷综合_中| 久久99精品国产.久久久久久| 国产一区二区三区久久久 | 床上的激情91.| av在线一区二区| 欧美二区在线| 亚洲国产高清一区二区三区| 在线看片日韩| 日韩天天综合| 久久久久成人精品免费播放动漫| 一道本成人在线| 欧美久久久影院| 日韩欧美一区在线| 国产欧美综合在线观看第十页| 亚洲欧美偷拍三级| 午夜精品福利久久久| 久久精品免费观看| 国产精品69久久久久水密桃|