com.univocity.parsers.common.input.concurrent.FixedInstancePool Maven / Gradle / Ivy
/*******************************************************************************
* Copyright 2014 uniVocity Software Pty Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
******************************************************************************/
package com.univocity.parsers.common.input.concurrent;
import java.util.*;
/**
* A very simple object instance pool with a fixed size.
*
* This is essentially an immutable circular queue. Elements are not added nor removed. Pointers to the head and tail of the queue identify what is the next available entry.
*
Use {@link FixedInstancePool#allocate()} to get an available {@link Entry} from the pool. If all objects are allocated then the thread will block until an element is released.
*
{@link FixedInstancePool#release(Entry)} releases an allocated {@link Entry} for reuse.
*
* @param the class of objects stored in the instance pool
*
* @author uniVocity Software Pty Ltd - [email protected]
* @see Entry
*/
abstract class FixedInstancePool {
final Entry[] instancePool;
private final int[] instanceIndexes;
private int head = 0;
private int tail = 0;
int count = 0;
private int lastInstanceIndex = 0;
/**
* Creates a new instance pool with the given size. Upon instantiation, the {@link FixedInstancePool#newInstance()} method will be called to fill in the instance pool, and the pool
* can then have its entries allocated for use (and reuse).
*
* @param size the size of the fixed instance pool.
*/
@SuppressWarnings("unchecked")
FixedInstancePool(int size) {
instancePool = new Entry[size];
instanceIndexes = new int[size];
Arrays.fill(instanceIndexes, -1);
instancePool[0] = new Entry(newInstance(), 0);
instanceIndexes[0] = 0;
}
/**
* Creates a new instance of the given type of objects stored as entries of this instance pool
* This method is called in the constructor of this class for initialization of the instance array and must always return a new instance.
*
* @return returns a new instance to use in the pool
*/
protected abstract T newInstance();
/**
* Retrieves the next available entry in this instance pool. Blocks until an entry becomes available (through {@link FixedInstancePool#release(Entry)}).
*
* @return the next available entry in this instance pool
*/
public synchronized Entry allocate() {
while (count == instancePool.length) {
try {
wait(50);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return new Entry(newInstance(), -1);
}
}
int index = instanceIndexes[head];
if (index == -1) {
index = ++lastInstanceIndex;
instanceIndexes[index] = index;
instancePool[index] = new Entry(newInstance(), index);
}
Entry out = instancePool[index];
// instanceIndexes[head] = -1; //enable to print the queue's contents for debugging purposes
head++;
if (head == instancePool.length) {
head = 0;
}
count++;
return out;
}
/**
* Releases the given entry and makes it available for allocation (by {@link FixedInstancePool#allocate()})
*
* @param e the entry to be released and made available for reuse.
*/
public synchronized void release(Entry e) {
if (e.index != -1) {
instanceIndexes[tail++] = e.index;
if (tail == instancePool.length) {
tail = 0;
}
count--;
}
notify();
}
}