Spring boot preloading data from database in bean

◇◆丶佛笑我妖孽 提交于 2019-12-06 13:40:52

问题


I'm asking for elements that have a recursive relationship in one table:

child id | parent id

1 | null

2 | 1

3 | 1

4 | 2

And so on...To ask for this whole datastructure it takes about 10 secondsAt the moment I can't redesign this table because it cost too much time (for more information: Spring Repository performance issues with recursive ORM Class)

Now, I am thinking about preloading all data during spring startup in a bean, so that the client "communicates" with the bean and I update the data in bean and database. I think it doesn't matter how long this startup takes, but it matters how long the user has to wait for its answer.

So far, I didn't manage it to preload it. I was trying to create a bean like this:

public class AppConfig extends WebMvcConfigurerAdapter {
...
@Autowired
SkillDAO skillDAO
...
@Bean(name="allSkills")
public List<Skill> allSkills(){
    return skillDAO.findBySkill(null);
}
...

It doesn't work cause I get an error:

Caused by: org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'dataSource': Requested bean is currently in creation: Is there an unresolvable circular reference?

By the way, I create all my beans in AppConfig. When I delete this "allSkills" bean, it works again.

UPDATE

@Component
public class MyListener {

@Autowired
private SkillDAO skillDAO;

@EventListener
public void onApplicationReady(ApplicationReadyEvent ready) {
    System.out.println("++++++++++++++ HALLO +++++++++++++");
    skillDAO.findBySkill(null);
}
}

...

@Bean(name="skillBean")
public MyListener myListener() {
    return new MyListener();
}

回答1:


A Cache would solve your requirement elegantly.

Ideally you would have a Service which has a method which returns the skills.

This method could looke like this:

import org.springframework.cache.annotation.Cacheable;

@Cacheable(value = "skills", key = "#root.methodName")
public List<Skill> getAllSkills() {
    return ... your skills ...;
}

To enable Caching in Spring Boot, add the @EnableCaching annotation to your configurations class, and add a Bean to configure it:

import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.cache.concurrent.ConcurrentMapCacheManager;

@Bean
public CacheManager cacheManager() {
    return new ConcurrentMapCacheManager("skills", "othercachename");
}

This way, the method getAllSkills is just executed once, the first time it's called, and afterwards the values are returned from the cache-manager without even calling the method.




回答2:


You can listen for the ApplicationReadyEvent, and then call the method (from @yglodt's answer) to initialise your cache.

Example code:

@Component
public class MyEventsListener {

    @Autowired
    SkillsService skillsService;

    @EventListener
    public void onApplicationReady(ApplicationReadyEvent ready) {
        skillsService.getAllSkills();
    }
}

Also remember that if you call getAllSkills() from within the SkillsService bean itself, you will not hit the cache, because the method is only advised when it is called on an injected proxy of the class.

If you're deploying the application as an executable jar (rather than a war file), then the simplest solution is to invoke the code that you want to run at start-up from your main method:

public class Application {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);
        SkillsService skillsService = context.getBean(SkillsService.class);
        skillsService.getAllSkills();
    }

}


来源:https://stackoverflow.com/questions/35427728/spring-boot-preloading-data-from-database-in-bean

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!