com.caucho.util.ConcurrentArrayList Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.util;
import java.lang.reflect.Array;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.AbstractCollection;
import java.util.ArrayList;
import java.util.Iterator;
/**
* A Concurrent array similar to the JDK's CopyOnWriteArrayList but
* saves a fixed array to avoid extra allocations.
*/
public class ConcurrentArrayList extends AbstractCollection {
private final Class> _type;
private final ArrayList _list = new ArrayList();
private E []_array;
public ConcurrentArrayList(Class type)
{
_type = type;
updateArray();
}
@Override
public int size()
{
return _array.length;
}
public E get(int index)
{
return _array[index];
}
@Override
public boolean contains(Object testValue)
{
for (E element : _array) {
if (element.equals(testValue))
return true;
}
return false;
}
public E find(K key, Match match)
{
for (E element : _array) {
if (match.isMatch(element, key))
return element;
}
return null;
}
public int indexOf(K key, Match match)
{
E []array = _array;
for (int i = 0; i < array.length; i++) {
if (match.isMatch(array[i], key)) {
return i;
}
}
return -1;
}
@Override
public boolean add(E value)
{
synchronized (_list) {
_list.add(value);
updateArray();
return true;
}
}
public E addIfAbsent(E value)
{
synchronized (_list) {
int index = _list.indexOf(value);
if (index >= 0)
return _list.get(index);
_list.add(value);
updateArray();
return null;
}
}
public E addIfAbsent(E value, Match match, K key)
{
synchronized (_list) {
E oldValue = find(key, match);
if (oldValue != null)
return oldValue;
add(value);
return null;
}
}
public E set(int index, E value)
{
synchronized (_list) {
while (_list.size() <= index) {
_list.add(null);
}
E oldValue = _list.set(index, value);
updateArray();
return oldValue;
}
}
@Override
public boolean remove(Object value)
{
synchronized (_list) {
boolean result = _list.remove(value);
updateArray();
return result;
}
}
public E remove(int index)
{
synchronized (_list) {
Object value = _list.remove(index);
updateArray();
return (E) value;
}
}
public E remove(K key, Match match)
{
synchronized (_list) {
int index = indexOf(key, match);
if (index < 0) {
return null;
}
E value = _array[index];
_list.remove(index);
updateArray();
return value;
}
}
@Override
public Iterator iterator()
{
return new ArrayIterator(_array);
}
@Override
public E []toArray()
{
return _array;
}
@SuppressWarnings("unchecked")
private void updateArray()
{
E []array = (E []) Array.newInstance(_type, _list.size());
_list.toArray(array);
_array = array;
}
public static Class> calculateType(Class> cl)
{
return calculateType(cl, cl);
}
public static Class> calculateType(Class> topClass, Class> cl)
{
if (cl == null)
throw new UnsupportedOperationException(topClass.toString());
Type type = cl.getGenericSuperclass();
if (type instanceof ParameterizedType) {
ParameterizedType pType = (ParameterizedType) type;
return (Class) pType.getActualTypeArguments()[0];
}
else {
return calculateType(topClass, cl.getSuperclass());
}
}
public interface Match {
public boolean isMatch(E element, K key);
}
public static class ArrayIterator implements Iterator {
private final E []_array;
int _index;
ArrayIterator(E []array)
{
_array = array;
}
@Override
public boolean hasNext()
{
return _index < _array.length;
}
@Override
public E next()
{
if (_index < _array.length)
return _array[_index++];
else
return null;
}
@Override
public void remove()
{
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy