我们在Spring Boot应用程序中使用EHCACHE。我们的春季启动版本是2.0.3.Release和Spring-boot-starter-Cache 2.0.3.Release使用EHCACHE 3.5.2。
我们使用Ehcache的动机是它既符合JSR107又提供了支持。
以下是我们的春季配置:
@Configuration
@ConditionalOnWebApplication
@EnableCaching
public class CacheConfig {
@Autowired
private ApplicationContext context;
@Bean
public JCacheManagerFactoryBean jCacheManagerFactoryBean() throws IOException {
JCacheManagerFactoryBean jCacheManagerFactoryBean = new JCacheManagerFactoryBean();
Resource resource = context.getResource("classpath:mts/ehcache.xml");
jCacheManagerFactoryBean.setCacheManagerUri(resource.getURI());
return jCacheManagerFactoryBean;
}
@Bean
public JCacheCacheManager ehCacheCacheManager() throws IOException {
Properties props = System.getProperties();
props.setProperty(Caching.JAVAX_CACHE_CACHING_PROVIDER, "org.ehcache.jsr107.EhcacheCachingProvider");
JCacheCacheManager jCacheCacheManager = new JCacheCacheManager();
jCacheCacheManager.setCacheManager(jCacheManagerFactoryBean().getObject());
jCacheCacheManager.setTransactionAware(true);
return jCacheCacheManager;
}
}
我们在生产中面临的问题是,对于磁盘持续的缓存,尺寸适中,我们在emoveAll操作中有以下Java.io.eofexception错误:
Error : RuntimeException: java.io.EOFException
java.lang.RuntimeException: java.io.EOFException
at org.terracotta.offheapstore.disk.storage.FileBackedStorageEngine$FileChunk.readKeyBuffer(FileBackedStorageEngine.java:541) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.disk.storage.FileBackedStorageEngine.readKeyBuffer(FileBackedStorageEngine.java:265) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.storage.PortabilityBasedStorageEngine.readKey(PortabilityBasedStorageEngine.java:119) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.OffHeapHashMap$DirectEntry.<init>(OffHeapHashMap.java:1540) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.OffHeapHashMap$EntryIterator.create(OffHeapHashMap.java:1518) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.OffHeapHashMap$EntryIterator.create(OffHeapHashMap.java:1511) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.OffHeapHashMap$HashIterator.next(OffHeapHashMap.java:1407) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.AbstractLockedOffHeapHashMap$LockedEntryIterator.next(AbstractLockedOffHeapHashMap.java:399) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.AbstractLockedOffHeapHashMap$LockedEntryIterator.next(AbstractLockedOffHeapHashMap.java:392) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.terracotta.offheapstore.concurrent.AbstractConcurrentOffHeapMap$AggregateIterator.next(AbstractConcurrentOffHeapMap.java:553) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.ehcache.impl.internal.store.offheap.AbstractOffHeapStore$1.next(AbstractOffHeapStore.java:499) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.ehcache.impl.internal.store.offheap.AbstractOffHeapStore$1.next(AbstractOffHeapStore.java:489) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.ehcache.core.EhcacheBase$Jsr107CacheBase.removeAll(EhcacheBase.java:708) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at org.ehcache.jsr107.Eh107Cache.removeAll(Eh107Cache.java:304) ~[ehcache-3.5.2.jar!/:3.5.2 7941fa2573343b31ae56a12564404552c6d6eff0]
at com.mycompany.myproject.services.cache.service.impl.CacheService.doClearCacheWithName(CacheService.java:56) ~[MyProjectServicesCache_classes.jar!/:?]
在调用removeAll操作的代码中没有什么特别的。只需获取名称的缓存,然后呼叫全部清除:
private void doClearCacheWithName(String cacheName) {
Cache<Object, Object> cache = cacheManager.getCache(cacheName);
if (cache == null) {
throw new MyProjectException(String.format("Cache with name : %s does not exist!", cacheName));
}
logger.info(String.format("Clearing cache with name : %s", cacheName));
cache.removeAll();
}
这是ourbigcache的生产配置:
<cache alias="ourBigCache">
<expiry>
<ttl unit="seconds">21600</ttl>
</expiry>
<resources>
<heap unit="entries">1000</heap>
<disk unit="MB">4096</disk>
</resources>
</cache>
我们无法在本地或测试环境中重现。
请注意,此缓存的用法很高(阅读计数的生产非常高),但我想这应该没有任何区别。
我找不到任何类似报告的问题。有一些非常古老的磁盘问题,但它们太老且不相似:
https://sourceforge.net/p/ehcache/discussion/322278/thread/e7a62df3/http://forums.terracotta.org/forums/posts/list/2694.page
任何帮助将不胜感激。
问候
从我对ehcache-users的答复中复制:
很可能这只是https://github.com/ehcache/ehcache3/issues/2542的另一种症状,该症状已在74239A93E14EB747777841FFFA36C971EF中固定。
不幸的是,此修复程序尚未在任何地方合并。如果您想拿起它,您可以剪裁自己的主人(这可能是一件好事)。否则,您必须等待3.7行上的第一个点释放(此时未知时间表)。
chris