0

我正在使用 Spring SpEL 评估一些结果,我想缓存这些结果,因此我不必评估具有相同参数的表达式。

我缓存的密钥对象:

@Data
@AllArgsConstructor
public class CachedResult {
    private String baseName;
    private Interval interval;

    public boolean isBetweenInclusive(final DateTime date) {
        return interval.contains(date) || interval.getEnd().isEqual(date);
    }
}

我找到interval包含给定记录的解决方案dateTime

public String getEvaluatedResult(final String baseName, final DateTime dateTime) {
    return cache.asMap().entrySet()
            .stream()
            .filter(entry -> entry.getKey().getBaseName().equals(baseName) && entry.getKey().isBetweenInclusive(dateTime))
            .findFirst()
            .map(Map.Entry::getValue)
            .orElse(null);
}

我想利用cache.get(key, valueLoader)方法,以便它可以在需要时将值本身放入缓存中,但我想不出一种方法来利用isBetweenInclusive这种方法。

我尝试在遇到问题时发表评论:

public class MyCache {

    private final Cache<CachedResult, String> cache;

    public DefaultRolloverCache(final int expiration) {
        this.cache = CacheBuilder.newBuilder()
                .expireAfterWrite(expiration, TimeUnit.MINUTES)
                .build();
    }

    public String getFileName(final String baseName, final DateTime dateTime, final Callable<String> valueLoader) {
        try {
            return cache.get(new CachedResult(baseName, null/*How to find an interval that covers the given dateTime?*/), valueLoader);
        } catch (final ExecutionException e) {
            throw new IllegalArgumentException(String.format("Cannot read fileName from cache with basename: '%s' and dateTime: %s", baseName, dateTime), e);
        }
    }
}

我将此方法称为:

cache.getFileName(baseName, new DateTime(), () -> doSlowCalculations(baseName));

当然,由于我不知道如何使用上面提到的方法,所以我必须自己使用cache.put(new CachedResult(...))将记录放入缓存中。

asMap有没有比像地图一样调用和过滤缓存更好的方法来过滤缓存?我可以以某种方式使用cache.get(key, valueLoader)Guavas 甚至 GuavasCacheLoader以便它可以自动输入值吗?

随着性能的发展,我一次最多可以有 5-10 条记录,但是我会从中读取很多,所以读取时间对我来说非常重要,我不确定我当前的实现是否会迭代5-10 条记录,并检查每条记录是最好的方法。

4

1 回答 1

0

阅读路易斯写道的评论后,我的最终解决方案:

public String getFileName(final String baseName, final DateTime dateTime, final Supplier<String> valueLoader) {
    final Optional<String> cached = cache.asMap().entrySet()
            .stream()
            .filter(entry -> entry.getKey().getBaseName().equals(baseName) && entry.getKey().isBetweenInclusive(dateTime))
            .findFirst()
            .map(Map.Entry::getValue);

    if (cached.isPresent()) {
        return cached.get();
    } else {
        final String evaluatedValue = valueLoader.get();
        cache.put(new CachedResult(baseName, intervalCalculator.getInterval(dateTime)), evaluatedValue);
        return evaluatedValue;
    }
}
于 2017-01-27T09:14:52.700 回答