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
.
组成键的元素需要有正确的 equals
和 hashcode
定义,这样即使你最终得到不同的实例,相同的值也会导致正确的缓存命中.
我正在尝试缓存我的结果,但每次调用页面(刷新)都会命中数据库。感谢任何帮助,谢谢。
控制器
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
.
组成键的元素需要有正确的 equals
和 hashcode
定义,这样即使你最终得到不同的实例,相同的值也会导致正确的缓存命中.