1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.contacts.util; 18 19 import com.android.contacts.test.NeededForTesting; 20 21 import android.util.LruCache; 22 23 import java.util.concurrent.atomic.AtomicInteger; 24 25 import javax.annotation.concurrent.Immutable; 26 import javax.annotation.concurrent.ThreadSafe; 27 28 /** 29 * An LRU cache in which all items can be marked as expired at a given time and it is possible to 30 * query whether a particular cached value is expired or not. 31 * <p> 32 * A typical use case for this is caching of values which are expensive to compute but which are 33 * still useful when out of date. 34 * <p> 35 * Consider a cache for contact information: 36 * <pre>{@code 37 * private ExpirableCache<String, Contact> mContactCache;}</pre> 38 * which stores the contact information for a given phone number. 39 * <p> 40 * When we need to store contact information for a given phone number, we can look up the info in 41 * the cache: 42 * <pre>{@code 43 * CachedValue<Contact> cachedContact = mContactCache.getCachedValue(phoneNumber); 44 * }</pre> 45 * We might also want to fetch the contact information again if the item is expired. 46 * <pre> 47 * if (cachedContact.isExpired()) { 48 * fetchContactForNumber(phoneNumber, 49 * new FetchListener() { 50 * @Override 51 * public void onFetched(Contact contact) { 52 * mContactCache.put(phoneNumber, contact); 53 * } 54 * }); 55 * }</pre> 56 * and insert it back into the cache when the fetch completes. 57 * <p> 58 * At a certain point we want to expire the content of the cache because we know the content may 59 * no longer be up-to-date, for instance, when resuming the activity this is shown into: 60 * <pre> 61 * @Override 62 * protected onResume() { 63 * // We were paused for some time, the cached value might no longer be up to date. 64 * mContactCache.expireAll(); 65 * super.onResume(); 66 * } 67 * </pre> 68 * The values will be still available from the cache, but they will be expired. 69 * <p> 70 * If interested only in the value itself, not whether it is expired or not, one should use the 71 * {@link #getPossiblyExpired(Object)} method. If interested only in non-expired values, one should 72 * use the {@link #get(Object)} method instead. 73 * <p> 74 * This class wraps around an {@link LruCache} instance: it follows the {@link LruCache} behavior 75 * for evicting items when the cache is full. It is possible to supply your own subclass of LruCache 76 * by using the {@link #create(LruCache)} method, which can define a custom expiration policy. 77 * Since the underlying cache maps keys to cached values it can determine which items are expired 78 * and which are not, allowing for an implementation that evicts expired items before non expired 79 * ones. 80 * <p> 81 * This class is thread-safe. 82 * 83 * @param <K> the type of the keys 84 * @param <V> the type of the values 85 */ 86 @ThreadSafe 87 public class ExpirableCache<K, V> { 88 /** 89 * A cached value stored inside the cache. 90 * <p> 91 * It provides access to the value stored in the cache but also allows to check whether the 92 * value is expired. 93 * 94 * @param <V> the type of value stored in the cache 95 */ 96 public interface CachedValue<V> { 97 /** Returns the value stored in the cache for a given key. */ getValue()98 public V getValue(); 99 100 /** 101 * Checks whether the value, while still being present in the cache, is expired. 102 * 103 * @return true if the value is expired 104 */ isExpired()105 public boolean isExpired(); 106 } 107 108 /** 109 * Cached values storing the generation at which they were added. 110 */ 111 @Immutable 112 private static class GenerationalCachedValue<V> implements ExpirableCache.CachedValue<V> { 113 /** The value stored in the cache. */ 114 public final V mValue; 115 /** The generation at which the value was added to the cache. */ 116 private final int mGeneration; 117 /** The atomic integer storing the current generation of the cache it belongs to. */ 118 private final AtomicInteger mCacheGeneration; 119 120 /** 121 * @param cacheGeneration the atomic integer storing the generation of the cache in which 122 * this value will be stored 123 */ GenerationalCachedValue(V value, AtomicInteger cacheGeneration)124 public GenerationalCachedValue(V value, AtomicInteger cacheGeneration) { 125 mValue = value; 126 mCacheGeneration = cacheGeneration; 127 // Snapshot the current generation. 128 mGeneration = mCacheGeneration.get(); 129 } 130 131 @Override getValue()132 public V getValue() { 133 return mValue; 134 } 135 136 @Override isExpired()137 public boolean isExpired() { 138 return mGeneration != mCacheGeneration.get(); 139 } 140 } 141 142 /** The underlying cache used to stored the cached values. */ 143 private LruCache<K, CachedValue<V>> mCache; 144 145 /** 146 * The current generation of items added to the cache. 147 * <p> 148 * Items in the cache can belong to a previous generation, but in that case they would be 149 * expired. 150 * 151 * @see ExpirableCache.CachedValue#isExpired() 152 */ 153 private final AtomicInteger mGeneration; 154 ExpirableCache(LruCache<K, CachedValue<V>> cache)155 private ExpirableCache(LruCache<K, CachedValue<V>> cache) { 156 mCache = cache; 157 mGeneration = new AtomicInteger(0); 158 } 159 160 /** 161 * Returns the cached value for the given key, or null if no value exists. 162 * <p> 163 * The cached value gives access both to the value associated with the key and whether it is 164 * expired or not. 165 * <p> 166 * If not interested in whether the value is expired, use {@link #getPossiblyExpired(Object)} 167 * instead. 168 * <p> 169 * If only wants values that are not expired, use {@link #get(Object)} instead. 170 * 171 * @param key the key to look up 172 */ getCachedValue(K key)173 public CachedValue<V> getCachedValue(K key) { 174 return mCache.get(key); 175 } 176 177 /** 178 * Returns the value for the given key, or null if no value exists. 179 * <p> 180 * When using this method, it is not possible to determine whether the value is expired or not. 181 * Use {@link #getCachedValue(Object)} to achieve that instead. However, if using 182 * {@link #getCachedValue(Object)} to determine if an item is expired, one should use the item 183 * within the {@link CachedValue} and not call {@link #getPossiblyExpired(Object)} to get the 184 * value afterwards, since that is not guaranteed to return the same value or that the newly 185 * returned value is in the same state. 186 * 187 * @param key the key to look up 188 */ getPossiblyExpired(K key)189 public V getPossiblyExpired(K key) { 190 CachedValue<V> cachedValue = getCachedValue(key); 191 return cachedValue == null ? null : cachedValue.getValue(); 192 } 193 194 /** 195 * Returns the value for the given key only if it is not expired, or null if no value exists or 196 * is expired. 197 * <p> 198 * This method will return null if either there is no value associated with this key or if the 199 * associated value is expired. 200 * 201 * @param key the key to look up 202 */ 203 @NeededForTesting get(K key)204 public V get(K key) { 205 CachedValue<V> cachedValue = getCachedValue(key); 206 return cachedValue == null || cachedValue.isExpired() ? null : cachedValue.getValue(); 207 } 208 209 /** 210 * Puts an item in the cache. 211 * <p> 212 * Newly added item will not be expired until {@link #expireAll()} is next called. 213 * 214 * @param key the key to look up 215 * @param value the value to associate with the key 216 */ put(K key, V value)217 public void put(K key, V value) { 218 mCache.put(key, newCachedValue(value)); 219 } 220 221 /** 222 * Mark all items currently in the cache as expired. 223 * <p> 224 * Newly added items after this call will be marked as not expired. 225 * <p> 226 * Expiring the items in the cache does not imply they will be evicted. 227 */ expireAll()228 public void expireAll() { 229 mGeneration.incrementAndGet(); 230 } 231 232 /** 233 * Creates a new {@link CachedValue} instance to be stored in this cache. 234 * <p> 235 * Implementation of {@link LruCache#create(K)} can use this method to create a new entry. 236 */ newCachedValue(V value)237 public CachedValue<V> newCachedValue(V value) { 238 return new GenerationalCachedValue<V>(value, mGeneration); 239 } 240 241 /** 242 * Creates a new {@link ExpirableCache} that wraps the given {@link LruCache}. 243 * <p> 244 * The created cache takes ownership of the cache passed in as an argument. 245 * 246 * @param <K> the type of the keys 247 * @param <V> the type of the values 248 * @param cache the cache to store the value in 249 * @return the newly created expirable cache 250 * @throws IllegalArgumentException if the cache is not empty 251 */ create(LruCache<K, CachedValue<V>> cache)252 public static <K, V> ExpirableCache<K, V> create(LruCache<K, CachedValue<V>> cache) { 253 return new ExpirableCache<K, V>(cache); 254 } 255 256 /** 257 * Creates a new {@link ExpirableCache} with the given maximum size. 258 * 259 * @param <K> the type of the keys 260 * @param <V> the type of the values 261 * @return the newly created expirable cache 262 */ create(int maxSize)263 public static <K, V> ExpirableCache<K, V> create(int maxSize) { 264 return create(new LruCache<K, CachedValue<V>>(maxSize)); 265 } 266 } 267