0

我正在使用咖啡因缓存并寻找一种方法来更新缓存中的值而不更改其过期时间。

场景是我使用缓存来加速数据加载。5 秒的数据更改延迟是可以接受的,但我预计返回速度会很快。此外,我希望这些缓存在第一次命中后 1 天到期,以避免不必要的内存使用。

因此,我希望每个缓存的键都持续一天,但它的值每 5 秒更新一次。

refreshAfterWrite方法似乎很接近,但刷新持续时间后它的第一个返回值仍然是旧的。这对我来说并不理想,因为两次点击之间的持续时间可能是几个小时。在这种情况下,我仍然想要一个相对较新的结果(不超过 5 秒)。

所以我试图手动更新每个键。

首先,我以这种方式构建了一个具有 24 小时过期时间的缓存:

cache = Caffeine.newBuilder()
    .expireAfterWrite(24, TimeUnit.HOURS)
    .build();

然后我每 5 秒编写一个计划任务,迭代缓存中的键并执行以下操作:

cache.asMap().computeIfPresent(key, mapperFunction);

然后我检查了钥匙的年龄:

cache.policy().expireAfterWrite().get().ageOf(key)

然而,年龄并没有像预期的那样增长。我认为该computeIfPresent 方法被视为“写入”,因此到期时间也会更新。

有没有办法在不改变咖啡因过期时间的情况下进行价值更新?或者我的场景的任何其他方法?

4

1 回答 1

0

写入是映射的创建或更新,因此expireAfterWrite不适合您。相反,您可以设置自定义过期策略,该策略设置初始持续时间并且在读取或更新时不执行任何操作。这是使用 完成的expireAfter(Expiry),例如

LoadingCache<Key, Graph> graphs = Caffeine.newBuilder()
    .expireAfter(new Expiry<Key, Graph>() {
      public long expireAfterCreate(Key key, Graph graph, long currentTime) {
        return TimeUnit.HOURS.toNanos(24);
      }
      public long expireAfterUpdate(Key key, Graph graph, 
          long currentTime, long currentDuration) {
        return currentDuration;
      }
      public long expireAfterRead(Key key, Graph graph,
          long currentTime, long currentDuration) {
        return currentDuration;
      }
    })
    .build(key -> createExpensiveGraph(key));
于 2021-01-27T17:40:04.443 回答