Springboot @Value注入boolean設(shè)置默認(rèn)值方式
@Value注入boolean設(shè)置默認(rèn)值
問題描述
Springboot 中讀取配置文件
test:
業(yè)務(wù)代碼如下
@Value("${test:true}")
private boolean test;報錯如下
nested exception is org.springframework.beans.TypeMismatchException: Failed to convert value of type 'java.lang.String' to required type 'boolean'; nested exception is java.lang.IllegalArgumentException: Invalid boolean value []
問題分析
根據(jù)報錯可知,主要問題在于 注入時 test 的值是 String 類型,無法轉(zhuǎn)換成 boolean 類型。
@Value("${test:true}")
private String test;于是更改了接收類型,看看獲取到的值是否是 true,結(jié)果發(fā)現(xiàn) test 值為 “”,而不是設(shè)置的默認(rèn)值
解決方案
報錯問題在于只要配置文件中有 test: 所以系統(tǒng)就默認(rèn) test 為 “” 而不是按照我所設(shè)想的為空所以默認(rèn)值為 true。
直接刪除配置文件中的 test: 即可正常啟動。
@Value 源碼閱讀
在排查問題的過程中也粗略的跟讀了一下源碼
//org.springframework.beans.TypeConverterSupport#doConvert()
private <T> T doConvert(Object value, Class<T> requiredType, MethodParameter methodParam, Field field) throws TypeMismatchException {
? ? ?try {
? ? ? ? ?return field != null ? this.typeConverterDelegate.convertIfNecessary(value, requiredType, field) : this.typeConverterDelegate.convertIfNecessary(value, requiredType, methodParam);
? ? ?} catch (ConverterNotFoundException var6) {
? ? ? ? ?throw new ConversionNotSupportedException(value, requiredType, var6);
? ? ?} catch (ConversionException var7) {
? ? ? ? ?throw new TypeMismatchException(value, requiredType, var7);
? ? ?} catch (IllegalStateException var8) {
? ? ? ? ?throw new ConversionNotSupportedException(value, requiredType, var8);
? ? ?} catch (IllegalArgumentException var9) {
? ? ?// 最終異常從這里拋出
? ? ? ? ?throw new TypeMismatchException(value, requiredType, var9);
? ? ?}
?}最終賦值在
//org.springframework.beans.TypeConverterDelegate#doConvertTextValue()
private Object doConvertTextValue(Object oldValue, String newTextValue, PropertyEditor editor) {
? ? try {
? ? ? ? editor.setValue(oldValue);
? ? } catch (Exception var5) {
? ? ? ? if (logger.isDebugEnabled()) {
? ? ? ? ? ? logger.debug("PropertyEditor [" + editor.getClass().getName() + "] does not support setValue call", var5);
? ? ? ? }
? ? }
?? ?// 此處發(fā)現(xiàn) newTextValue 為 ""
? ? editor.setAsText(newTextValue);
? ? return editor.getValue();
}接下來就是如何將 字符串 true 轉(zhuǎn)換為 boolean 的具體代碼:
// org.springframework.beans.propertyeditors.CustomBooleanEditor#setAsText()
? ? public void setAsText(String text) throws IllegalArgumentException {
? ? ? ? String input = text != null ? text.trim() : null;
? ? ? ? if (this.allowEmpty && !StringUtils.hasLength(input)) {
? ? ? ? ? ? this.setValue((Object)null);
? ? ? ? } else if (this.trueString != null && this.trueString.equalsIgnoreCase(input)) {
? ? ? ? ? ? this.setValue(Boolean.TRUE);
? ? ? ? } else if (this.falseString != null && this.falseString.equalsIgnoreCase(input)) {
? ? ? ? ? ? this.setValue(Boolean.FALSE);
? ? ? ? } else if (this.trueString != null || !"true".equalsIgnoreCase(input) && !"on".equalsIgnoreCase(input) && !"yes".equalsIgnoreCase(input) && !"1".equals(input)) {
? ? ? ? ? ? if (this.falseString != null || !"false".equalsIgnoreCase(input) && !"off".equalsIgnoreCase(input) && !"no".equalsIgnoreCase(input) && !"0".equals(input)) {
? ? ? ? ? ? ? ? throw new IllegalArgumentException("Invalid boolean value [" + text + "]");
? ? ? ? ? ? }
? ? ? ? ? ? this.setValue(Boolean.FALSE);
? ? ? ? } else {
? ? ? ? ? ? this.setValue(Boolean.TRUE);
? ? ? ? }
? ? }tips:windows 中使用 IDEA 去查找類可以使用 ctrl + shift +alt +N的快捷鍵組合去查詢,mac 系統(tǒng)則是 commond + O
Spring解析@Value
1、初始化PropertyPlaceholderHelper對象
? ? protected String placeholderPrefix = "${";
?
?? ?protected String placeholderSuffix = "}";
?? ?@Nullable
?? ?protected String valueSeparator = ":";?
private static final Map<String, String> wellKnownSimplePrefixes = new HashMap<>(4);
?
?? ?static {
?? ??? ?wellKnownSimplePrefixes.put("}", "{");
?? ??? ?wellKnownSimplePrefixes.put("]", "[");
?? ??? ?wellKnownSimplePrefixes.put(")", "(");
?? ?}
?
public PropertyPlaceholderHelper(String placeholderPrefix, String placeholderSuffix,
?? ??? ??? ?@Nullable String valueSeparator, boolean ignoreUnresolvablePlaceholders) {
?
?? ??? ?Assert.notNull(placeholderPrefix, "'placeholderPrefix' must not be null");
?? ??? ?Assert.notNull(placeholderSuffix, "'placeholderSuffix' must not be null");
? ? ? ? //默認(rèn)值${
?? ??? ?this.placeholderPrefix = placeholderPrefix;
? ? ? ? //默認(rèn)值}
?? ??? ?this.placeholderSuffix = placeholderSuffix;
?? ??? ?String simplePrefixForSuffix = wellKnownSimplePrefixes.get(this.placeholderSuffix);
? ? ? ? //當(dāng)前綴為空或跟定義的不匹配,取傳入的前綴
?? ??? ?if (simplePrefixForSuffix != null && this.placeholderPrefix.endsWith(simplePrefixForSuffix)) {
?? ??? ??? ?this.simplePrefix = simplePrefixForSuffix;
?? ??? ?}
?? ??? ?else {
?? ??? ??? ?this.simplePrefix = this.placeholderPrefix;
?? ??? ?}
? ? ? ? //默認(rèn)值:
?? ??? ?this.valueSeparator = valueSeparator;
?? ??? ?this.ignoreUnresolvablePlaceholders = ignoreUnresolvablePlaceholders;
?? ?}2、解析@Value
protected String parseStringValue(
?? ??? ??? ?String value, PlaceholderResolver placeholderResolver, Set<String> visitedPlaceholders) {
?
?? ??? ?StringBuilder result = new StringBuilder(value);
? ? ? ? //是否包含前綴,返回第一個前綴的開始index
?? ??? ?int startIndex = value.indexOf(this.placeholderPrefix);
?? ??? ?while (startIndex != -1) {
? ? ? ? ? ? //找到最后一個后綴的index
?? ??? ??? ?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");
?? ??? ??? ??? ?}
?? ??? ??? ??? ?// 遞歸判斷是否存在占位符,可以這樣寫${acm.endpoint:${address.server.domain:}}
?? ??? ??? ??? ?placeholder = parseStringValue(placeholder, placeholderResolver, visitedPlaceholders);
?? ??? ??? ??? ?// 根據(jù)key獲取對應(yīng)的值
?? ??? ??? ??? ?String propVal = placeholderResolver.resolvePlaceholder(placeholder);
? ? ? ? ? ? ? ? // 值不存在,但存在默認(rèn)值的分隔符
?? ??? ??? ??? ?if (propVal == null && this.valueSeparator != null) {
? ? ? ? ? ? ? ? ? ? // 獲取默認(rèn)值的索引
?? ??? ??? ??? ??? ?int separatorIndex = placeholder.indexOf(this.valueSeparator);
?? ??? ??? ??? ??? ?if (separatorIndex != -1) {
? ? ? ? ? ? ? ? ? ? ? ? // 切掉默認(rèn)值的字符串
?? ??? ??? ??? ??? ??? ?String actualPlaceholder = placeholder.substring(0, separatorIndex);
? ? ? ? ? ? ? ? ? ? ? ? // 切出默認(rèn)值
?? ??? ??? ??? ??? ??? ?String defaultValue = placeholder.substring(separatorIndex + this.valueSeparator.length());
? ? ? ? ? ? ? ? ? ? ? ? // 根據(jù)新的key獲取對應(yīng)的值
?? ??? ??? ??? ??? ??? ?propVal = placeholderResolver.resolvePlaceholder(actualPlaceholder);
? ? ? ? ? ? ? ? ? ? ? ? // 如果值不存在,則把默認(rèn)值賦值給當(dāng)前值
?? ??? ??? ??? ??? ??? ?if (propVal == null) {
?? ??? ??? ??? ??? ??? ??? ?propVal = defaultValue;
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ?}
? ? ? ? ? ? ? ? // 如果當(dāng)前值不為NULL
?? ??? ??? ??? ?if (propVal != null) {
?? ??? ??? ??? ??? ?// 遞歸獲取存在占位符的值信息
?? ??? ??? ??? ??? ?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 if (this.ignoreUnresolvablePlaceholders) {
?? ??? ??? ??? ??? ?// Proceed with unprocessed value.
?? ??? ??? ??? ??? ?startIndex = result.indexOf(this.placeholderPrefix, endIndex + this.placeholderSuffix.length());
?? ??? ??? ??? ?}
?? ??? ??? ??? ?else {
?? ??? ??? ??? ??? ?throw new IllegalArgumentException("Could not resolve placeholder '" +
?? ??? ??? ??? ??? ??? ??? ?placeholder + "'" + " in value \"" + value + "\"");
?? ??? ??? ??? ?}
?? ??? ??? ??? ?visitedPlaceholders.remove(originalPlaceholder);
?? ??? ??? ?}
?? ??? ??? ?else {
?? ??? ??? ??? ?startIndex = -1;
?? ??? ??? ?}
?? ??? ?}?
?? ??? ?return result.toString();
?? ?}以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
詳談java中int和Integer的區(qū)別及自動裝箱和自動拆箱
這篇文章主要介紹了詳談java中int和Integer的區(qū)別及自動裝箱和自動拆箱,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
java中map和對象互轉(zhuǎn)工具類的實現(xiàn)示例
這篇文章主要介紹了java中map和對象互轉(zhuǎn)工具類的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
java 中如何獲取字節(jié)碼文件的相關(guān)內(nèi)容
這篇文章主要介紹了java 中如何獲取字節(jié)碼文件的相關(guān)內(nèi)容的相關(guān)資料,需要的朋友可以參考下2017-04-04
SpringBoot使用@Async注解可能會遇到的8大坑點匯總
SpringBoot中,@Async注解可以實現(xiàn)異步線程調(diào)用,用法簡單,體驗舒適,但是你一定碰到過異步調(diào)用不生效的情況,今天,我就列出90%的人都可能會遇到的8大坑點,需要的朋友可以參考下2023-09-09

