简体   繁体   中英

Use application.properties in a non-spring injected class

I have a class which is created with new B(this); in this class BI want to use a value from the application.properties. But because (as far as I understand) because it's not created with Spring it won't have any injection (I use the @Value annotation)

That is how the class is created:

@Component
public class A {

    public B getB(){return new B(this);}

    /**
        a lot more stuff
    **/
}

The class in question:

public class B {

    private A a;

    public B(A a){
        this.a = a;
    }

    @Value("${threshold}")
    private String threshold;  //this is null because it's not created with Spring

    public String getThreshold(){
        return threshold;
    }
    /**
        a lot more stuff
    **/

}

So my question is the following:

1) How can I use the value in the application.properties file or

2) How can I write B that it is created with Spring?

Some background information:

  • I didn't wrote the initial code so I don't want to change too much but want to modify it so it can be maintained better in the future
  • I don't have that much Spring knowledge and only start getting more and more familiar with it.
  • In point 2) I'm struggling because of the constructorand how to set it via Spring...

Any help would be appreciated

Here's an example of using @ConfigurationProperties to bind your application.properties to a POJO.

Say you have a application.properties file like this,

mail.hostname=mailer@mail.com
mail.port=9000

You can create a POJO like this for the above scenario.

( Note: If your spring boot version is lower than 2.2 you might want to annotate this class with @Configuration and @Component as well)

@ConfigurationProperties(prefix = "mail")
public class ConfigProperties {

    private String hostname;
    private String port;

    // Create Getters and Setters

}

When the spring boot application initializes, it will automatically map values from application.properties into this POJO. If you want to use this, all you have to do is create a variable and mark it with @Autowire

@Component
public class TestClass {

    @Autowire
    private ConfigProperties properties;

    public void printSomething() {
       System.println(properties.getHostname());
       System.println(properties.getPort());
    }

}

Following up on your question...

Since the class in your example is not managed by spring, and you have to keep it this way for some reason, you can use the following helper class to manually autowire a spring bean in a non spring managed class.

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Service;

@Service
public class BeanUtil implements ApplicationContextAware {

    private static ApplicationContext context;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        context = applicationContext;
    }

    /**
     *Use this method to manually autowire Spring Beans into classes that are not managed by Spring.
     *
     *@param beanClass - Class type of the required bean.
     **/
    public static <T> T getBean(Class<T> beanClass) {
        return context.getBean(beanClass);
    }

}

To use this in your class B do the following.

public class B {

    private ConfigProperties properties;

    public B() {
        BeanUtil.getBean(ConfigProperties.class); // This will initialize properties variable properly.
    }

    public void printSomething() {
       System.println(properties.getHostname());
       System.println(properties.getPort());
    }

}

You can write as below:

@Component
public class A {

    @Value("${threshold}")
    private String threshold;

    public B getB(){
        return new B(this);
    }

    public String getThreshold() {
        return threshold;
    }

}

public class B {

    private A a;
    private String threshold;

    public B(A a){
        this.a = a;
        this.threshold=a.getThreshold();
    }

    public String getThreshold(){
        return threshold;
    }

}

You can create a class that will load the properties for you directly from a file. The file must be in the resources package.

Config properties class where these properties are defined

@ConfigurationProperties(prefix = "reader")
public class ReaderProperties {
    private String threshold;
}

The property class

import java.util.Properties;

class MyProperties {

    private final Properties props;

    private MyProperties(Class<?> propertiesClass) throws IOException {
        this.props = new Properties();
        this.props.load(propertiesClass.getResourceAsStream("/application.properties"));
    }
    
    public String getThreshold() {
        return props.getProperty("threshold");
    }

}

Then inside B:

public class B {

    private A a;
    private String threshold;

    public B(A a){
        this.a = a;
        MyProperties props = new MyProperties(ReaderProperties.class);
        this.threshold = props.getThreshold();
    }

    public String getThreshold(){
        return threshold;
    }
    /**
        a lot more stuff
    **/

}

The existing answers work if you're willing and able to annotate class B with @ConfigurationProperties but this might not be feasible if B is provided as part of non-Spring library, and or if B needs to be used in non-Spring projects.

Fortunately Spring does provide a more portable way to do this if B is a POJO/JavaBean:

Example class B :

public class B {
    private String threshold;
    // Is a POJO/JavaBean i.e. has default constructor, getters, setters etc.
}

In application.properties :

my.config.prefix.threshold=42

Define a Spring bean in a @Configuration class for your Spring Boot project eg:

@Configuration
public class ApplicationConfig {
    @Bean
    @ConfigurationProperties("my.config.prefix")
    public B getB() {
        // Spring will later use setters to configure the fields per my.config.prefix
        return new B();
    }
}

You can now rely on Spring to autowire the dependency eg using field injection 1 :

@Component
public class A {
    @Autowired
    private B b; // b.getThreshold() == 42
}

Unless a class is strictly for running or configuring a Spring Boot application, I would err to towards keeping the code more portable by defining a bean.


1 Field injection is generally not recommended but it's used here for concision.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM