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

gw.util.concurrent.LockingLazyVar Maven / Gradle / Ivy

There is a newer version: 1.18.2
Show newest version
/*
 * Copyright 2014 Guidewire Software, Inc.
 */

package gw.util.concurrent;

import gw.lang.reflect.TypeSystem;

import java.util.concurrent.locks.Lock;

public abstract class LockingLazyVar
{
  protected final static Object NULL = new Object();
  private volatile T _val = null;
  private final Lock _lock;

  /**
   * Constructs a LockingLazyVar that will use itself as the object of synchronization.
   */
  public LockingLazyVar()
  {
    _lock = TypeSystem.getGlobalLock();
  }

  /**
   * Constructs a LockingLazyVar that will synchronize on the given object.
   */
  protected LockingLazyVar(Lock lock)
  {
    _lock = lock;
  }

  /**
   * @return the value of this lazy var, created if necessary
   */
  public final T get()
  {
    T result = _val;
    if(result == NULL) {
      return null;
    }
    if( result == null )
    {
      _lock.lock();
      try{

        result = _val;
        if(result == NULL) {
          return null;
        }
        if( result == null )
        {
          result = init();

          //The extra space makes all the difference

          if (result == null) {
            _val = (T)NULL;
          } else {
            _val = result;
          }
        }
      } finally {
        _lock.unlock();
      }
    }
    return result;
  }

  protected abstract T init();

  /**
   * Clears the variable, forcing the next call to {@link #get()} to re-calculate
   * the value.
   */
  public final T clear()
  {
    T hold;
    _lock.lock();
    try
    {
      hold = _val;
      _val = null;
    } finally {
      _lock.unlock();
    }
    return hold;
  }

  public final void clearNoLock()
  {
    _val = null;
  }

  protected void initDirectly( T val )
  {
    _lock.lock();
    try
    {
      _val = val;
    }
    finally
    {
      _lock.unlock();
    }
  }

  public boolean isLoaded() {
    return _val != null;
  }

  /**
   * A simple init interface to make LockingLazyVar's easier to construct
   * from gosu.
   */
  public static interface LazyVarInit {
    public Q init();
  }

  /**
   * Creates a new LockingLazyVar based on the type of the LazyVarInit passed in.
   * This method is intended to be called with blocks from Gosu.
   */
  public static  LockingLazyVar make( final LazyVarInit init ) {
    return new LockingLazyVar(){
      protected Q init()
      {
        return init.init();
      }
    };
  }

  public static  LockingLazyVar make( Lock lock, final LazyVarInit init ) {
    return new LockingLazyVar( lock ) {
      protected Q init()
      {
        return init.init();
      }
    };
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy