Spring Ehcache 仅在同一请求中调用方法时才工作

Spring Ehcache works only if method called in same request

我正在尝试缓存我的结果,但每次调用页面(刷新)都会命中数据库。感谢任何帮助,谢谢。

控制器

showPage(){
    MyPage<PhoneInfo> myPage = (MyPage<PhoneInfo>) phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>(page.intValue(), size.intValue()));
    //further experiment.. below does not hit db
    myPage = (MyPage<PhoneInfo>) phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>(page.intValue(), size.intValue()));
}

服务方法的第二次调用没有进行数据库调用,这是正确的。 然而,页面刷新再次调用 showPage() 并且第一次调用服务再次命中数据库。我希望它来自缓存。

我测试了 2 个单元测试,两个测试都调用了相同的服务方法,测试 1 命中数据库而测试 2 跳过,这是正确的。测试中的多个服务调用也被跳过,这也是正确的。

规格

Spring 4.3.3.RELEASE, Security 4.2.3.RELEASE, Hibernate 5.2.8.Final, Ehcache 2.10.4

缓存配置

@Configuration
@EnableCaching
public class CachingConfig {
    @Bean
    public CacheManager cacheManager(net.sf.ehcache.CacheManager cacheManager) {
        return new EhCacheCacheManager(cacheManager);
    }

    @Bean
    public EhCacheManagerFactoryBean ehCacheCacheManager() {
        EhCacheManagerFactoryBean cmfb = new EhCacheManagerFactoryBean();
        cmfb.setConfigLocation(new ClassPathResource("ehcache.xml"));
        cmfb.setShared(true);
        return cmfb;
    }
}

src/resources/ehcache.xml

<?xml version="1.0" encoding="UTF-8"?> <ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   xsi:noNamespaceSchemaLocation="ehcache.xsd"     updateCheck="true"  monitoring="autodetect"     dynamicConfig="true">

    <diskStore path="java.io.tmpdir" />

    <cache name="myPageCache"       maxEntriesLocalHeap="1000"      maxEntriesLocalDisk="10000"         eternal="false"         diskSpoolBufferSizeMB="20"      timeToIdleSeconds="86400" timeToLiveSeconds="86400"         overflowToDisk="false"      memoryStoreEvictionPolicy="LFU"         transactionalMode="off">        <persistence strategy="localTempSwap" />    </cache>

</ehcache>

更新 1 下面列出服务方法

Cacheable(value = "myPageCache", key = "{#options, #pageable}")
public MyPage<PhoneInfo>findAllByOption(PhoneOption options, MyPage<PhoneInfo> pageable){
    MyPage<PhoneInfo> myPage = phoneInfoRepoImpl.findAll(options, pageable);
    myPage.setTotalRows(_phoneInfoService.getCount(options, pageable));
    return myPage;
}

@Cacheable(value = "myPageCache", key = "#options")
public int getCount(PhoneOption options, MyPage<PhoneInfo> pageable){
    return phoneInfoRepoImpl.getCount(options, pageable);
}

更新 2 单元测试

@Test
public void test1(){
    PhoneOption options = new PhoneOption();
    options.setName("239");
    options.setStatus('1');
    MyPage<PhoneInfo> myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>()); //hits db, Ok
    myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>()); //skips db, OK
    myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>(2, 20)); //hits db, OK as MyPage changed
}

@Test
public void test2(){
    PhoneOption options = new PhoneOption();
    options.setName("239");
    options.setStatus('1');
    MyPage<PhoneInfo> myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>()); //these options exactly same as Test1, skips db, OK!
    myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>()); //skips db, Ok
    myPage = phoneInfoService.findAllByOption(options, new MyPage<PhoneInfo>(2, 20)); //skips db, Ok
}

更新 3在所有测试对象中实现了 hashCode 和 equals

现在在单元测试中它总是命中数据库!所以行为变得更糟。 equal/same 对象的 hashcode 相同,equals 也能正常工作,对于具有相似属性的对象返回 true。

您的键 "{#options, #pageable}" 每次都会有新值,即使您在这些对象中传递了相同的一组值,因为它们在每次页面刷新后都是全新的对象。因此,它们每次在第一次调用时都会被缓存。为避免这种情况,您需要更改 key.Try 使用对象内部的字段,例如 #object.field.

组成键的元素需要有正确的 equalshashcode 定义,这样即使你最终得到不同的实例,相同的值也会导致正确的缓存命中.