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

com.barchart.http.server.ObjectPool Maven / Gradle / Ivy

package com.barchart.http.server;

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.Callable;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * A pool of reusable objects which creates new instances on demand.
 * 
 * @param 
 *            The poolable object type
 */
public class ObjectPool {

	// TODO replace with something more efficient
	/* The backing queue */
	private final BlockingQueue objectPool;

	/* User-provided object creator callback */
	private final Callable objectCreator;

	/* Number of pool objects that have been created */
	private final AtomicInteger created = new AtomicInteger(0);

	/* Maximum size of this pool */
	private final int maxObjects;

	/**
	 * Create an unbounded object pool.
	 */
	public ObjectPool(final Callable creator_) {
		maxObjects = -1;
		objectCreator = creator_;
		objectPool = new LinkedBlockingQueue();
	}

	/**
	 * Create a fixed-size object pool.
	 */
	public ObjectPool(final int maxObjects_, final Callable creator_) {
		maxObjects = maxObjects_;
		objectCreator = creator_;
		objectPool = new ArrayBlockingQueue(maxObjects);
	}

	/**
	 * Take an object from the pool if available.
	 * 
	 * @return The pooled object, or null if none are available
	 */
	public T poll() {
		final T obj = getOrCreate();
		if (obj != null) {
			return obj;
		}
		return objectPool.poll();
	}

	/**
	 * Take an object from the pool, waiting the specified time for one to
	 * become available.
	 * 
	 * @param timeout
	 *            The time to wait
	 * @param units
	 *            The time units
	 * @return The pooled object
	 * @throws InterruptedException
	 *             If the thread is interrupted while waiting for an object
	 */
	public T poll(final long timeout, final TimeUnit units)
			throws InterruptedException {
		final T obj = getOrCreate();
		if (obj != null) {
			return obj;
		}
		return objectPool.poll(timeout, units);
	}

	/**
	 * Take an object from the pool, blocking until one becomes available.
	 * 
	 * @return The pooled object
	 * @throws InterruptedException
	 *             If the thread is interrupted while waiting for an object
	 */
	public T take() throws InterruptedException {
		final T obj = getOrCreate();
		if (obj != null) {
			return obj;
		}
		return objectPool.take();
	}

	/**
	 * Get an object from the pool. If none are available and the pool is not
	 * full, create a new object and return it.
	 * 
	 * @return The pooled object, or null if none are available and the pool is
	 *         full
	 */
	protected T getOrCreate() {

		// First just try to pull an object off the queue
		final T instance = objectPool.poll();

		if (instance == null) {

			int count = created.get();

			// Try to increase the counter as long as the pool is not full
			while (maxObjects == -1 || count < maxObjects) {

				if (created.compareAndSet(count, count + 1)) {

					try {

						// If successful, we can create a new object.
						return objectCreator.call();

					} catch (final Exception e) {

						// Create failed, restore counter
						created.decrementAndGet();

						throw new RuntimeException(
								"Unhandled exception in object creator", e);

					}

				}

				// compareAndSet() failed, get new size and try again until the
				// pool is full
				count = created.get();

			}

		}

		return instance;

	}

	/**
	 * Return an object to the pool.
	 * 
	 * @param object
	 */
	public void give(final T object) {
		if (!objectPool.offer(object)) {
			throw new IllegalStateException(
					"Attempted to return an object to a full pool. "
							+ "Only return objects created by this pool.");
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy