java – 确保始终从Spring应用程序中的缓存中检索数据

在使用Spring Framework构建的Java EE应用程序中,我需要在DAO中执行一些非常昂贵的操作,这可能需要几分钟的时间.使用Spring MVC,当用户的请求映射到控制器方法时,我通过我的Controller命中DAO:

@RequestMapping(value = "/categories.do")
public ModelAndView categories(
        @PathVariable("cc") String cc,
        @PathVariable("ll") String ll) {
    Locale locale = new Locale(ll, cc);
    ModelAndView result = getView("categories", locale);
    Map<Category, List<Product>> allProducts = supplyDao.getAllProducts(locale);
    result.addObject("products", allProducts);
    return result;
}

getAllProducts方法向外部Web服务发出大量请求,以绘制产品所需的所有数据.该方法通过Spring的@Cacheable注释缓存在底层的Ehcache实现上:

@Cacheable(value = CACHE_NAME, key = CACHE_KEY_PREFIX + "'(' + #p0 + ')'")
public Map<Category, List<Product>> getAllProducts(Locale locale) {
        // a lot of HTTP requests firing from here
}

这种方法的问题在于,当缓存为空时,页面基本上是不可访问的.此外,如果在缓存为空时多个请求命中页面,则DAO方法将再次触发,并且所有请求将并行重复.据我所知,第二个问题的解决方案是使用BlockingCache,但我还没有机会实现它.

我想要的是让控制器方法始终从缓存中提取结果.我想实现一个触发所有语言环境的缓存填充的@PostConstruct方法.就像是:

@PostConstruct
public void populateCaches() {
    for (Locale locale : localeList) {
        getAllProducts(locale);
    }
}

我不介意初始人口需要一段时间,因为服务器很少重启.我还将缓存过期时间设置为三天 – 数据不会经常更新,并且没有服务最新版本的危险.

我想要做的是按设定的时间间隔运行TimerTask,比方说两天23小时,这将迫使DAO方法从Web服务中提取所有产品数据.然后,该数据将替换缓存中的数据,而不会过期.然后将重置缓存到期计数器 – 数据将在三天后再次到期.这样,控制器方法总是从缓存中获取产品数据,并且页面将是响应的.

我的问题是,如果我使用Spring’s cache abstraction,我将如何实施这样的方法?我需要在我的方法中直接处理CacheManager吗?

另一个问题是:我正确地解决了这个问题吗?有没有更好的方法呢?

解决方法:

阻止呼叫,直到缓存可用

@Cacheable(cacheName="yourCache", decoratedCacheType= DecoratedCacheType.SELF_POPULATING_CACHE)
public List<String> getWhatever(int id) {
//call database
}

您可以使用自动刷新缓存而不是timertask:

@Cacheable(cacheName="yourCache", refreshInterval=1000, decoratedCacheType= DecoratedCacheType.REFRESHING_SELF_POPULATING_CACHE)
public List<String> getWhatever(int id) {
  //call database
}

kudos

上一篇:Spring 4.2.4(不使用spring boot)EhCache 3 Hibernate 4.2.1


下一篇:Hazelcast与Ehcache Hazelcast使用Spring和Hibernate在service / dao层缓存注释?