com.github.benmanes.caffeine.cache.LocalLoadingCache Maven / Gradle / Ivy
/*
* Copyright 2015 Ben Manes. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.github.benmanes.caffeine.cache;
import static java.util.Objects.requireNonNull;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.function.Function;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* This class provides a skeletal implementation of the {@link LoadingCache} interface to minimize
* the effort required to implement a {@link LocalCache}.
*
* @author [email protected] (Ben Manes)
*/
interface LocalLoadingCache, K, V>
extends LocalManualCache, LoadingCache {
Logger logger = Logger.getLogger(LocalLoadingCache.class.getName());
/** Returns the {@link CacheLoader} used by this cache. */
CacheLoader super K, V> cacheLoader();
/** Returns the {@link CacheLoader} as a mapping function. */
Function mappingFunction();
/** Returns whether the cache loader supports bulk loading. */
boolean hasBulkLoader();
/** Returns whether the supplied cache loader has bulk load functionality. */
default boolean hasLoadAll(CacheLoader super K, V> loader) {
try {
Method classLoadAll = loader.getClass().getMethod("loadAll", Iterable.class);
Method defaultLoadAll = CacheLoader.class.getMethod("loadAll", Iterable.class);
return !classLoadAll.equals(defaultLoadAll);
} catch (NoSuchMethodException | SecurityException e) {
logger.log(Level.WARNING, "Cannot determine if CacheLoader can bulk load", e);
return false;
}
}
@Override
default V get(K key) {
return cache().computeIfAbsent(key, mappingFunction());
}
@Override
default Map getAll(Iterable extends K> keys) {
return hasBulkLoader() ? loadInBulk(keys) : loadSequentially(keys);
}
/** Sequentially loads each missing entry. */
default Map loadSequentially(Iterable extends K> keys) {
int count = 0;
Map result = new HashMap<>();
Iterator extends K> iter = keys.iterator();
while (iter.hasNext()) {
K key = iter.next();
count++;
try {
V value = get(key);
if (value != null) {
result.put(key, value);
}
} catch (Throwable t) {
int remaining;
if (keys instanceof Collection>) {
remaining = ((Collection>) keys).size() - count;
} else {
remaining = 0;
while (iter.hasNext()) {
remaining++;
iter.next();
}
}
cache().statsCounter().recordMisses(remaining);
throw t;
}
}
return Collections.unmodifiableMap(result);
}
/** Batch loads the missing entries. */
default Map loadInBulk(Iterable extends K> keys) {
Map found = cache().getAllPresent(keys);
Set keysToLoad = new HashSet<>();
for (K key : keys) {
if (!found.containsKey(key)) {
keysToLoad.add(key);
}
}
if (keysToLoad.isEmpty()) {
return found;
}
Map result = new HashMap<>(found);
bulkLoad(keysToLoad, result);
return Collections.unmodifiableMap(result);
}
/**
* Performs a non-blocking bulk load of the missing keys. Any missing entry that materializes
* during the load are replaced when the loaded entries are inserted into the cache.
*/
default void bulkLoad(Set keysToLoad, Map result) {
boolean success = false;
long startTime = cache().statsTicker().read();
try {
@SuppressWarnings("unchecked")
Map loaded = (Map) cacheLoader().loadAll(keysToLoad);
loaded.forEach((key, value) -> {
cache().put(key, value, /* notifyWriter */ false);
if (keysToLoad.contains(key)) {
result.put(key, value);
}
});
success = !loaded.isEmpty();
} catch (RuntimeException e) {
throw e;
} catch (Exception e) {
throw new CompletionException(e);
} finally {
long loadTime = cache().statsTicker().read() - startTime;
if (success) {
cache().statsCounter().recordLoadSuccess(loadTime);
} else {
cache().statsCounter().recordLoadFailure(loadTime);
}
}
}
@Override
default void refresh(K key) {
requireNonNull(key);
long[] writeTime = new long[1];
long startTime = cache().statsTicker().read();
V oldValue = cache().getIfPresentQuietly(key, writeTime);
CompletableFuture refreshFuture = (oldValue == null)
? cacheLoader().asyncLoad(key, cache().executor())
: cacheLoader().asyncReload(key, oldValue, cache().executor());
refreshFuture.whenComplete((newValue, error) -> {
long loadTime = cache().statsTicker().read() - startTime;
if (error != null) {
logger.log(Level.WARNING, "Exception thrown during refresh", error);
cache().statsCounter().recordLoadFailure(loadTime);
return;
}
boolean[] discard = new boolean[1];
cache().compute(key, (k, currentValue) -> {
if (currentValue == null) {
return newValue;
} else if (currentValue == oldValue) {
long expectedWriteTime = writeTime[0];
if (cache().hasWriteTime()) {
cache().getIfPresentQuietly(key, writeTime);
}
if (writeTime[0] == expectedWriteTime) {
return newValue;
}
}
discard[0] = true;
return currentValue;
}, /* recordMiss */ false, /* recordLoad */ false);
if (discard[0] && cache().hasRemovalListener()) {
cache().notifyRemoval(key, newValue, RemovalCause.REPLACED);
}
if (newValue == null) {
cache().statsCounter().recordLoadFailure(loadTime);
} else {
cache().statsCounter().recordLoadSuccess(loadTime);
}
});
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy