001/* 002 * Copyright (C) 2007 The Guava Authors 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except 005 * in compliance with the License. You may obtain a copy of the License at 006 * 007 * http://www.apache.org/licenses/LICENSE-2.0 008 * 009 * Unless required by applicable law or agreed to in writing, software distributed under the License 010 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express 011 * or implied. See the License for the specific language governing permissions and limitations under 012 * the License. 013 */ 014 015package com.google.common.base; 016 017import static com.google.common.base.Internal.toNanosSaturated; 018import static com.google.common.base.NullnessCasts.uncheckedCastNullableTToT; 019import static com.google.common.base.Preconditions.checkArgument; 020import static com.google.common.base.Preconditions.checkNotNull; 021 022import com.google.common.annotations.GwtCompatible; 023import com.google.common.annotations.GwtIncompatible; 024import com.google.common.annotations.J2ktIncompatible; 025import com.google.common.annotations.VisibleForTesting; 026import java.io.IOException; 027import java.io.ObjectInputStream; 028import java.io.Serializable; 029import java.time.Duration; 030import java.util.concurrent.TimeUnit; 031import org.jspecify.annotations.Nullable; 032 033/** 034 * Useful suppliers. 035 * 036 * <p>All methods return serializable suppliers as long as they're given serializable parameters. 037 * 038 * @author Laurence Gonsalves 039 * @author Harry Heymann 040 * @since 2.0 041 */ 042@GwtCompatible(emulated = true) 043public final class Suppliers { 044 private Suppliers() {} 045 046 /** 047 * Returns a new supplier which is the composition of the provided function and supplier. In other 048 * words, the new supplier's value will be computed by retrieving the value from {@code supplier}, 049 * and then applying {@code function} to that value. Note that the resulting supplier will not 050 * call {@code supplier} or invoke {@code function} until it is called. 051 */ 052 public static <F extends @Nullable Object, T extends @Nullable Object> Supplier<T> compose( 053 Function<? super F, T> function, Supplier<F> supplier) { 054 return new SupplierComposition<>(function, supplier); 055 } 056 057 private static class SupplierComposition<F extends @Nullable Object, T extends @Nullable Object> 058 implements Supplier<T>, Serializable { 059 final Function<? super F, T> function; 060 final Supplier<F> supplier; 061 062 SupplierComposition(Function<? super F, T> function, Supplier<F> supplier) { 063 this.function = checkNotNull(function); 064 this.supplier = checkNotNull(supplier); 065 } 066 067 @Override 068 @ParametricNullness 069 public T get() { 070 return function.apply(supplier.get()); 071 } 072 073 @Override 074 public boolean equals(@Nullable Object obj) { 075 if (obj instanceof SupplierComposition) { 076 SupplierComposition<?, ?> that = (SupplierComposition<?, ?>) obj; 077 return function.equals(that.function) && supplier.equals(that.supplier); 078 } 079 return false; 080 } 081 082 @Override 083 public int hashCode() { 084 return Objects.hashCode(function, supplier); 085 } 086 087 @Override 088 public String toString() { 089 return "Suppliers.compose(" + function + ", " + supplier + ")"; 090 } 091 092 private static final long serialVersionUID = 0; 093 } 094 095 /** 096 * Returns a supplier which caches the instance retrieved during the first call to {@code get()} 097 * and returns that value on subsequent calls to {@code get()}. See: <a 098 * href="http://en.wikipedia.org/wiki/Memoization">memoization</a> 099 * 100 * <p>The returned supplier is thread-safe. The delegate's {@code get()} method will be invoked at 101 * most once unless the underlying {@code get()} throws an exception. The supplier's serialized 102 * form does not contain the cached value, which will be recalculated when {@code get()} is called 103 * on the deserialized instance. 104 * 105 * <p>When the underlying delegate throws an exception then this memoizing supplier will keep 106 * delegating calls until it returns valid data. 107 * 108 * <p>If {@code delegate} is an instance created by an earlier call to {@code memoize}, it is 109 * returned directly. 110 */ 111 public static <T extends @Nullable Object> Supplier<T> memoize(Supplier<T> delegate) { 112 if (delegate instanceof NonSerializableMemoizingSupplier 113 || delegate instanceof MemoizingSupplier) { 114 return delegate; 115 } 116 return delegate instanceof Serializable 117 ? new MemoizingSupplier<T>(delegate) 118 : new NonSerializableMemoizingSupplier<T>(delegate); 119 } 120 121 @VisibleForTesting 122 static class MemoizingSupplier<T extends @Nullable Object> implements Supplier<T>, Serializable { 123 private transient Object lock = new Object(); 124 125 final Supplier<T> delegate; 126 transient volatile boolean initialized; 127 // "value" does not need to be volatile; visibility piggy-backs 128 // on volatile read of "initialized". 129 transient @Nullable T value; 130 131 MemoizingSupplier(Supplier<T> delegate) { 132 this.delegate = checkNotNull(delegate); 133 } 134 135 @Override 136 @ParametricNullness 137 // We set the field only once (during construction or deserialization). 138 @SuppressWarnings("SynchronizeOnNonFinalField") 139 public T get() { 140 // A 2-field variant of Double Checked Locking. 141 if (!initialized) { 142 synchronized (lock) { 143 if (!initialized) { 144 T t = delegate.get(); 145 value = t; 146 initialized = true; 147 return t; 148 } 149 } 150 } 151 // This is safe because we checked `initialized`. 152 return uncheckedCastNullableTToT(value); 153 } 154 155 @Override 156 public String toString() { 157 return "Suppliers.memoize(" 158 + (initialized ? "<supplier that returned " + value + ">" : delegate) 159 + ")"; 160 } 161 162 @GwtIncompatible // serialization 163 @J2ktIncompatible // serialization 164 private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { 165 in.defaultReadObject(); 166 lock = new Object(); 167 } 168 169 private static final long serialVersionUID = 0; 170 } 171 172 @VisibleForTesting 173 static class NonSerializableMemoizingSupplier<T extends @Nullable Object> implements Supplier<T> { 174 private final Object lock = new Object(); 175 176 @SuppressWarnings("UnnecessaryLambda") // Must be a fixed singleton object 177 private static final Supplier<@Nullable Void> SUCCESSFULLY_COMPUTED = 178 () -> { 179 throw new IllegalStateException(); // Should never get called. 180 }; 181 182 private volatile Supplier<T> delegate; 183 // "value" does not need to be volatile; visibility piggy-backs on volatile read of "delegate". 184 private @Nullable T value; 185 186 NonSerializableMemoizingSupplier(Supplier<T> delegate) { 187 this.delegate = checkNotNull(delegate); 188 } 189 190 @Override 191 @ParametricNullness 192 @SuppressWarnings("unchecked") // Cast from Supplier<Void> to Supplier<T> is always valid 193 public T get() { 194 // Because Supplier is read-heavy, we use the "double-checked locking" pattern. 195 if (delegate != SUCCESSFULLY_COMPUTED) { 196 synchronized (lock) { 197 if (delegate != SUCCESSFULLY_COMPUTED) { 198 T t = delegate.get(); 199 value = t; 200 delegate = (Supplier<T>) SUCCESSFULLY_COMPUTED; 201 return t; 202 } 203 } 204 } 205 // This is safe because we checked `delegate`. 206 return uncheckedCastNullableTToT(value); 207 } 208 209 @Override 210 public String toString() { 211 Supplier<T> delegate = this.delegate; 212 return "Suppliers.memoize(" 213 + (delegate == SUCCESSFULLY_COMPUTED 214 ? "<supplier that returned " + value + ">" 215 : delegate) 216 + ")"; 217 } 218 } 219 220 /** 221 * Returns a supplier that caches the instance supplied by the delegate and removes the cached 222 * value after the specified time has passed. Subsequent calls to {@code get()} return the cached 223 * value if the expiration time has not passed. After the expiration time, a new value is 224 * retrieved, cached, and returned. See: <a 225 * href="http://en.wikipedia.org/wiki/Memoization">memoization</a> 226 * 227 * <p>The returned supplier is thread-safe. The supplier's serialized form does not contain the 228 * cached value, which will be recalculated when {@code get()} is called on the reserialized 229 * instance. The actual memoization does not happen when the underlying delegate throws an 230 * exception. 231 * 232 * <p>When the underlying delegate throws an exception then this memoizing supplier will keep 233 * delegating calls until it returns valid data. 234 * 235 * @param duration the length of time after a value is created that it should stop being returned 236 * by subsequent {@code get()} calls 237 * @param unit the unit that {@code duration} is expressed in 238 * @throws IllegalArgumentException if {@code duration} is not positive 239 * @since 2.0 240 */ 241 @SuppressWarnings("GoodTime") // Prefer the Duration overload 242 public static <T extends @Nullable Object> Supplier<T> memoizeWithExpiration( 243 Supplier<T> delegate, long duration, TimeUnit unit) { 244 checkNotNull(delegate); 245 checkArgument(duration > 0, "duration (%s %s) must be > 0", duration, unit); 246 return new ExpiringMemoizingSupplier<>(delegate, unit.toNanos(duration)); 247 } 248 249 /** 250 * Returns a supplier that caches the instance supplied by the delegate and removes the cached 251 * value after the specified time has passed. Subsequent calls to {@code get()} return the cached 252 * value if the expiration time has not passed. After the expiration time, a new value is 253 * retrieved, cached, and returned. See: <a 254 * href="http://en.wikipedia.org/wiki/Memoization">memoization</a> 255 * 256 * <p>The returned supplier is thread-safe. The supplier's serialized form does not contain the 257 * cached value, which will be recalculated when {@code get()} is called on the reserialized 258 * instance. The actual memoization does not happen when the underlying delegate throws an 259 * exception. 260 * 261 * <p>When the underlying delegate throws an exception then this memoizing supplier will keep 262 * delegating calls until it returns valid data. 263 * 264 * @param duration the length of time after a value is created that it should stop being returned 265 * by subsequent {@code get()} calls 266 * @throws IllegalArgumentException if {@code duration} is not positive 267 * @since 33.1.0 268 */ 269 @J2ktIncompatible 270 @GwtIncompatible // java.time.Duration 271 @SuppressWarnings("Java7ApiChecker") // no more dangerous that wherever the user got the Duration 272 @IgnoreJRERequirement 273 public static <T extends @Nullable Object> Supplier<T> memoizeWithExpiration( 274 Supplier<T> delegate, Duration duration) { 275 checkNotNull(delegate); 276 // The alternative of `duration.compareTo(Duration.ZERO) > 0` causes J2ObjC trouble. 277 checkArgument( 278 !duration.isNegative() && !duration.isZero(), "duration (%s) must be > 0", duration); 279 return new ExpiringMemoizingSupplier<>(delegate, toNanosSaturated(duration)); 280 } 281 282 @VisibleForTesting 283 @SuppressWarnings("GoodTime") // lots of violations 284 static class ExpiringMemoizingSupplier<T extends @Nullable Object> 285 implements Supplier<T>, Serializable { 286 private transient Object lock = new Object(); 287 288 final Supplier<T> delegate; 289 final long durationNanos; 290 transient volatile @Nullable T value; 291 // The special value 0 means "not yet initialized". 292 transient volatile long expirationNanos; 293 294 ExpiringMemoizingSupplier(Supplier<T> delegate, long durationNanos) { 295 this.delegate = delegate; 296 this.durationNanos = durationNanos; 297 } 298 299 @Override 300 @ParametricNullness 301 // We set the field only once (during construction or deserialization). 302 @SuppressWarnings("SynchronizeOnNonFinalField") 303 public T get() { 304 // Another variant of Double Checked Locking. 305 // 306 // We use two volatile reads. We could reduce this to one by 307 // putting our fields into a holder class, but (at least on x86) 308 // the extra memory consumption and indirection are more 309 // expensive than the extra volatile reads. 310 long nanos = expirationNanos; 311 long now = System.nanoTime(); 312 if (nanos == 0 || now - nanos >= 0) { 313 synchronized (lock) { 314 if (nanos == expirationNanos) { // recheck for lost race 315 T t = delegate.get(); 316 value = t; 317 nanos = now + durationNanos; 318 // In the very unlikely event that nanos is 0, set it to 1; 319 // no one will notice 1 ns of tardiness. 320 expirationNanos = (nanos == 0) ? 1 : nanos; 321 return t; 322 } 323 } 324 } 325 // This is safe because we checked `expirationNanos`. 326 return uncheckedCastNullableTToT(value); 327 } 328 329 @Override 330 public String toString() { 331 // This is a little strange if the unit the user provided was not NANOS, 332 // but we don't want to store the unit just for toString 333 return "Suppliers.memoizeWithExpiration(" + delegate + ", " + durationNanos + ", NANOS)"; 334 } 335 336 @GwtIncompatible // serialization 337 @J2ktIncompatible // serialization 338 private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { 339 in.defaultReadObject(); 340 lock = new Object(); 341 } 342 343 private static final long serialVersionUID = 0; 344 } 345 346 /** Returns a supplier that always supplies {@code instance}. */ 347 public static <T extends @Nullable Object> Supplier<T> ofInstance( 348 @ParametricNullness T instance) { 349 return new SupplierOfInstance<>(instance); 350 } 351 352 private static class SupplierOfInstance<T extends @Nullable Object> 353 implements Supplier<T>, Serializable { 354 @ParametricNullness final T instance; 355 356 SupplierOfInstance(@ParametricNullness T instance) { 357 this.instance = instance; 358 } 359 360 @Override 361 @ParametricNullness 362 public T get() { 363 return instance; 364 } 365 366 @Override 367 public boolean equals(@Nullable Object obj) { 368 if (obj instanceof SupplierOfInstance) { 369 SupplierOfInstance<?> that = (SupplierOfInstance<?>) obj; 370 return Objects.equal(instance, that.instance); 371 } 372 return false; 373 } 374 375 @Override 376 public int hashCode() { 377 return Objects.hashCode(instance); 378 } 379 380 @Override 381 public String toString() { 382 return "Suppliers.ofInstance(" + instance + ")"; 383 } 384 385 private static final long serialVersionUID = 0; 386 } 387 388 /** 389 * Returns a supplier whose {@code get()} method synchronizes on {@code delegate} before calling 390 * it, making it thread-safe. 391 */ 392 @J2ktIncompatible 393 public static <T extends @Nullable Object> Supplier<T> synchronizedSupplier( 394 Supplier<T> delegate) { 395 return new ThreadSafeSupplier<>(delegate); 396 } 397 398 @J2ktIncompatible 399 private static class ThreadSafeSupplier<T extends @Nullable Object> 400 implements Supplier<T>, Serializable { 401 final Supplier<T> delegate; 402 403 ThreadSafeSupplier(Supplier<T> delegate) { 404 this.delegate = checkNotNull(delegate); 405 } 406 407 @Override 408 @ParametricNullness 409 public T get() { 410 synchronized (delegate) { 411 return delegate.get(); 412 } 413 } 414 415 @Override 416 public String toString() { 417 return "Suppliers.synchronizedSupplier(" + delegate + ")"; 418 } 419 420 private static final long serialVersionUID = 0; 421 } 422 423 /** 424 * Returns a function that accepts a supplier and returns the result of invoking {@link 425 * Supplier#get} on that supplier. 426 * 427 * <p><b>Java 8+ users:</b> use the method reference {@code Supplier::get} instead. 428 * 429 * @since 8.0 430 */ 431 public static <T extends @Nullable Object> Function<Supplier<T>, T> supplierFunction() { 432 @SuppressWarnings("unchecked") // implementation is "fully variant" 433 SupplierFunction<T> sf = (SupplierFunction<T>) SupplierFunctionImpl.INSTANCE; 434 return sf; 435 } 436 437 private interface SupplierFunction<T extends @Nullable Object> extends Function<Supplier<T>, T> {} 438 439 private enum SupplierFunctionImpl implements SupplierFunction<@Nullable Object> { 440 INSTANCE; 441 442 // Note: This makes T a "pass-through type" 443 @Override 444 public @Nullable Object apply(Supplier<@Nullable Object> input) { 445 return input.get(); 446 } 447 448 @Override 449 public String toString() { 450 return "Suppliers.supplierFunction()"; 451 } 452 } 453}