如何在固定的超时时间内缓存单个对象是最佳方式?

70

谷歌的Guava库提供了CacheBuilder,允许创建带有过期键的ConcurrentHash,这样可以在固定超时后将条目删除。然而,我只需要缓存某种类型的单个实例。

使用Google Guava,缓存单个对象的最佳方法是什么?

1个回答

134

我会使用Guava的Suppliers.memoizeWithExpiration(Supplier delegate, long duration, TimeUnit unit)

public class JdkVersionService {

    @Inject
    private JdkVersionWebService jdkVersionWebService;

    // No need to check too often. Once a year will be good :) 
    private final Supplier<JdkVersion> latestJdkVersionCache
            = Suppliers.memoizeWithExpiration(jdkVersionSupplier(), 365, TimeUnit.DAYS);


    public JdkVersion getLatestJdkVersion() {
        return latestJdkVersionCache.get();
    }

    private Supplier<JdkVersion> jdkVersionSupplier() {
        return new Supplier<JdkVersion>() {
            public JdkVersion get() {
                return jdkVersionWebService.checkLatestJdkVersion();
            }
        };
    }
}

使用JDK 8更新

今天,我会使用JDK 8的方法引用和构造函数注入来编写更简洁的代码:

import java.util.concurrent.TimeUnit;
import java.util.function.Supplier;

import javax.inject.Inject;

import org.springframework.stereotype.Service;

import com.google.common.base.Suppliers;

@Service
public class JdkVersionService {

    private final Supplier<JdkVersion> latestJdkVersionCache;

    @Inject
    public JdkVersionService(JdkVersionWebService jdkVersionWebService) {
        this.latestJdkVersionCache = Suppliers.memoizeWithExpiration(
                jdkVersionWebService::checkLatestJdkVersion,
                365, TimeUnit.DAYS
        );
    }

    public JdkVersion getLatestJdkVersion() {
        return latestJdkVersionCache.get();
    }
}

你知道当供应商抛出一个已检查异常的情况下有什么解决方案吗? - user2417480
@user2417480 我可能会将该异常包装在 RuntimeException 中。如果您真的想保留已检查的异常,稍后可以取消包装它,但是在多个层之间携带已检查的异常通常是一种反模式...这就是为什么例如 spring-jdbc 具有转换层,将已检查的异常转换为运行时 DataAccessException。 - Etienne Neveu

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接