Spring启动不会替换 Spring 工具套件版本中的系统变量 {user.home}:3.8。4.RELEASE

SpringBoot don't replacen System variable {user.home} in Spring Tool Suite Version: 3.8.4.RELEASE

我使用 Spring Initializr 生成了一个 Spring Boot web 应用程序,在 macOS Sierra 中使用嵌入式 Tomcat + Thymeleaf 模板引擎。 我想在 Mac OS

中使用 系统变量 用户主文件夹名称

我的 Spring 引导应用程序中有此 Spring class 配置

@Configuration
@Profile("dev")
@PropertySource("file:///{user.home}/.devopsbuddy/application-dev.properties")
public class DevelopmentConfig {

    @Bean
    public EmailService emailService() {
        return new MockEmailService();
    }

}

但是我在启动应用程序时遇到了这个错误

Caused by: java.io.FileNotFoundException: /{user.home}/.devopsbuddy/application-dev.properties (No such file or directory)
    at java.io.FileInputStream.open0(Native Method)
    at java.io.FileInputStream.open(FileInputStream.java:195)
    at java.io.FileInputStream.<init>(FileInputStream.java:138)
    at java.io.FileInputStream.<init>(FileInputStream.java:93)
    at sun.net.www.protocol.file.FileURLConnection.connect(FileURLConnection.java:90)
    at sun.net.www.protocol.file.FileURLConnection.getInputStream(FileURLConnection.java:188)
    at org.springframework.core.io.UrlResource.getInputStream(UrlResource.java:169)
    at org.springframework.core.io.support.EncodedResource.getInputStream(EncodedResource.java:154)
    at org.springframework.core.io.support.PropertiesLoaderUtils.fillProperties(PropertiesLoaderUtils.java:98)
    at org.springframework.core.io.support.PropertiesLoaderUtils.fillProperties(PropertiesLoaderUtils.java:72)
    at org.springframework.core.io.support.PropertiesLoaderUtils.loadProperties(PropertiesLoaderUtils.java:58)
    at org.springframework.core.io.support.ResourcePropertySource.<init>(ResourcePropertySource.java:65)
    at org.springframework.core.io.support.DefaultPropertySourceFactory.createPropertySource(DefaultPropertySourceFactory.java:36)
    at org.springframework.context.annotation.ConfigurationClassParser.processPropertySource(ConfigurationClassParser.java:440)
    at org.springframework.context.annotation.ConfigurationClassParser.doProcessConfigurationClass(ConfigurationClassParser.java:271)
    at org.springframework.context.annotation.ConfigurationClassParser.processConfigurationClass(ConfigurationClassParser.java:245)
    at org.springframework.context.annotation.ConfigurationClassParser.parse(ConfigurationClassParser.java:190)
    at org.springframework.context.annotation.ConfigurationClassParser.doProcessConfigurationClass(ConfigurationClassParser.java:292)
    at org.springframework.context.annotation.ConfigurationClassParser.processConfigurationClass(ConfigurationClassParser.java:245)
    at org.springframework.context.annotation.ConfigurationClassParser.parse(ConfigurationClassParser.java:198)
    at org.springframework.context.annotation.ConfigurationClassParser.parse(ConfigurationClassParser.java:167)
    ... 13 common frames omitted

只要系统变量存在${sys:user.home}就是systax。

Spring 使用 PropertyPlaceholderConfigurer 解析所有占位符,即 ${variable or property name}。 所以,你应该使用 ${user.home}。请注意$标志。

@PropertySource 最初是作为 Spring 3.1 的一部分添加的,用于导入资源

@PropertySource Java doc中指出,

Any ${…} placeholders present in a @PropertySource resource location will be resolved against the set of property sources already registered against the environment.

对于你的问题,你没有添加美元符号($)。希望添加美元符号后,你的问题会得到解决。

您可以通过多种方式添加@PropertySource。对于基本使用,

@Configuration
@PropertySource(value = "classpath:application.properties")
public class ApplicationConfig {

    // more configuration ...
}

执行时,将从位于类路径根目录中的 application.properties 文件导入属性。类路径是默认位置,因此可以省略:

@Configuration
@PropertySource("application.properties")
public class ApplicationConfig {
}

或者,可以指定一个文件:位置来指定位于主机环境其他位置的属性文件:

@PropertySource("file:/path/to/application.properties")

或者您可以使用

@PropertySource("file:${CONF_DIR}/application.properties")
$ echo $CONF_DIR
/path/to/directory/with/app/config/files

在你的情况下,你必须使用`

@PropertySource("file:${user.home}/application.properties")

`

和 $user.home 将给出以下位置。您会将属性文件放在该位置。

 $ echo $user.home
 /home/yourusername

In Spring 4:

Spring 4 为@ProperySource 带来了两个新特性。

第一个新功能:

它处理丢失的文件。默认情况下,Spring 如果找不到已声明的文件,将抛出异常。

@PropertySource(value = "missing.properties", ignoreResourceNotFound = true)

如果不使用ignoreResourceNotFound = true,会报如下错误

java.lang.IllegalStateException: Failed to load ApplicationContext
[...]
Caused by: java.io.FileNotFoundException: class path resource [missing.properties] cannot be opened because it does not exist.

第二个新功能:

其次,有一个名为@PropertySources 的新注解,它允许您声明重复的@PropertySource 注解:

@PropertySources({
    @PropertySource("default.properties"),
    @PropertySource("overriding.properties")
})

N.B:同样,属性 文件声明的顺序很重要。如上例中的文件名所示,如果稍后声明的文件包含相同的键,则它们将覆盖任何先前的值。

放在一起

总而言之,属性 源配置可以像这样实现:

@Configuration
@PropertySources({
    @PropertySource("default.properties"),
    @PropertySource(value = "file:${CONF_DIR}/optional-override.properties", ignoreResourceNotFound = true)
}
public class ApplicationConfig {
}

In Java 8


在Java8 中,@PropertySources 注释将是多余的,因为Java8 引入了repeating annotation。这意味着可以在相同位置重复任意多的相同注释。

在Java8之前,要有一个重复的注解,必须将它们分组在一个容器注解中

@Manufactures({
@Manufacturer(name =”BMW”),
@Manufacturer(name = “Range Rover”)

})
public class Car{
//code goes in here
}

有了Java8repeating annotations,它给了我们写同样东西的灵活性,不需要任何容器注解

@Manufacturer(name = “BMW”)
@Manufacturer(name= “Range Rover”)
public class Car{
//code goes in here
}

虽然这里没有使用容器注释,但这次 Java 编译器负责将两个注释包装到一个容器中。

所有功劳归于 Mattias Severson

资源Link:

  1. Spring @PropertySource
  2. Java 8 Repeating Annotation Explained in 5 minutes