All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.bukkit.metadata.LazyMetadataValue Maven / Gradle / Ivy

There is a newer version: 2.4.0
Show newest version
package org.bukkit.metadata;

import org.apache.commons.lang.Validate;
import org.bukkit.plugin.Plugin;

import java.lang.ref.SoftReference;
import java.util.concurrent.Callable;

/**
 * The LazyMetadataValue class implements a type of metadata that is not
 * computed until another plugin asks for it.
 * 

* By making metadata values lazy, no computation is done by the providing * plugin until absolutely necessary (if ever). Additionally, * LazyMetadataValue objects cache their values internally unless overridden * by a {@link CacheStrategy} or invalidated at the individual or plugin * level. Once invalidated, the LazyMetadataValue will recompute its value * when asked. */ public class LazyMetadataValue extends MetadataValueAdapter implements MetadataValue { private static final Object ACTUALLY_NULL = new Object(); private Callable lazyValue; private CacheStrategy cacheStrategy; private SoftReference internalValue; /** * Initialized a LazyMetadataValue object with the default * CACHE_AFTER_FIRST_EVAL cache strategy. * * @param owningPlugin the {@link Plugin} that created this metadata * value. * @param lazyValue the lazy value assigned to this metadata value. */ public LazyMetadataValue(Plugin owningPlugin, Callable lazyValue) { this(owningPlugin, CacheStrategy.CACHE_AFTER_FIRST_EVAL, lazyValue); } /** * Initializes a LazyMetadataValue object with a specific cache strategy. * * @param owningPlugin the {@link Plugin} that created this metadata * value. * @param cacheStrategy determines the rules for caching this metadata * value. * @param lazyValue the lazy value assigned to this metadata value. */ public LazyMetadataValue(Plugin owningPlugin, CacheStrategy cacheStrategy, Callable lazyValue) { super(owningPlugin); Validate.notNull(cacheStrategy, "cacheStrategy cannot be null"); Validate.notNull(lazyValue, "lazyValue cannot be null"); this.internalValue = new SoftReference(null); this.lazyValue = lazyValue; this.cacheStrategy = cacheStrategy; } /** * Protected special constructor used by FixedMetadataValue to bypass * standard setup. * * @param owningPlugin the owning plugin */ protected LazyMetadataValue(Plugin owningPlugin) { super(owningPlugin); } public Object value() { eval(); Object value = internalValue.get(); if (value == ACTUALLY_NULL) { return null; } return value; } /** * Lazily evaluates the value of this metadata item. * * @throws MetadataEvaluationException if computing the metadata value * fails. */ private synchronized void eval() throws MetadataEvaluationException { if (cacheStrategy == CacheStrategy.NEVER_CACHE || internalValue.get() == null) { try { Object value = lazyValue.call(); if (value == null) { value = ACTUALLY_NULL; } internalValue = new SoftReference(value); } catch (Exception e) { throw new MetadataEvaluationException(e); } } } public synchronized void invalidate() { if (cacheStrategy != CacheStrategy.CACHE_ETERNALLY) { internalValue.clear(); } } /** * Describes possible caching strategies for metadata. */ public enum CacheStrategy { /** * Once the metadata value has been evaluated, do not re-evaluate the * value until it is manually invalidated. */ CACHE_AFTER_FIRST_EVAL, /** * Re-evaluate the metadata item every time it is requested */ NEVER_CACHE, /** * Once the metadata value has been evaluated, do not re-evaluate the * value in spite of manual invalidation. */ CACHE_ETERNALLY } }