Каков наилучший способ кэширования одного объекта в течение фиксированного времени ожидания?

В Google Guava есть CacheBuilder, который позволяет создавать ConcurrentHash с ключами с истекающим сроком действия, которые позволяют удалять записи после фиксированного тайм-аута. Однако мне нужно кэшировать только один экземпляр определенного типа.

Как лучше всего кэшировать один объект в течение фиксированного времени ожидания с помощью Google Guava?


person Alexey Zakharov    schedule 23.10.2011    source источник


Ответы (1)


Я бы использовал Suppliers.memoizeWithExpiration(делегат поставщика, большая продолжительность, единица измерения TimeUnit)

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();
    }
}
person Etienne Neveu    schedule 23.10.2011
comment
Знаете ли вы решение для случая, когда поставщик выдает проверенное исключение? - person user2417480; 23.09.2019
comment
@user2417480 user2417480 Я бы, наверное, обернул это исключение в RuntimeException. Возможно, разверните его позже, если вы действительно хотите сохранить проверенное исключение, но обычно это анти-шаблон для переноса проверенного исключения на несколько слоев... Вот почему, например. Spring-jdbc имеет уровень перевода для преобразования проверенных исключений в DataAccessException во время выполнения. - person Etienne Neveu; 23.01.2020