2009-03-15 12:02:36 -06:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 11:06:52 -07:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-07-21 21:47:29 -06:00
|
|
|
package java.util;
|
|
|
|
|
|
|
|
public class Collections {
|
2010-08-15 14:01:33 +02:00
|
|
|
|
2007-07-21 21:47:29 -06:00
|
|
|
private Collections() { }
|
|
|
|
|
2009-03-04 08:18:18 -07:00
|
|
|
public static void shuffle(List list, Random random) {
|
|
|
|
Object[] array = toArray(list, new Object[list.size()]);
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
int j = random.nextInt(array.length);
|
|
|
|
Object tmp = array[i];
|
|
|
|
array[i] = array[j];
|
|
|
|
array[j] = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
list.clear();
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
list.add(array[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void shuffle(List list) {
|
|
|
|
shuffle(list, new Random());
|
|
|
|
}
|
|
|
|
|
2008-02-28 11:37:10 -07:00
|
|
|
static <T> T[] toArray(Collection collection, T[] array) {
|
|
|
|
Class c = array.getClass().getComponentType();
|
|
|
|
|
2008-03-05 14:21:53 -07:00
|
|
|
if (array.length < collection.size()) {
|
2008-02-28 11:37:10 -07:00
|
|
|
array = (T[]) java.lang.reflect.Array.newInstance(c, collection.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
int i = 0;
|
|
|
|
for (Object o: collection) {
|
|
|
|
if (c.isInstance(o)) {
|
|
|
|
array[i++] = (T) o;
|
|
|
|
} else {
|
|
|
|
throw new ArrayStoreException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2007-10-16 19:17:37 -06:00
|
|
|
static String toString(Collection c) {
|
2007-08-23 19:57:42 -06:00
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
sb.append("[");
|
2007-10-16 19:17:37 -06:00
|
|
|
for (Iterator it = c.iterator(); it.hasNext();) {
|
2007-08-23 19:57:42 -06:00
|
|
|
sb.append(it.next());
|
|
|
|
if (it.hasNext()) {
|
|
|
|
sb.append(",");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sb.append("]");
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
static String toString(Map m) {
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
sb.append("{");
|
|
|
|
for (Iterator<Map.Entry> it = m.entrySet().iterator(); it.hasNext();) {
|
|
|
|
Map.Entry e = it.next();
|
|
|
|
sb.append(e.getKey())
|
|
|
|
.append("=")
|
|
|
|
.append(e.getValue());
|
|
|
|
if (it.hasNext()) {
|
|
|
|
sb.append(",");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sb.append("}");
|
|
|
|
return sb.toString();
|
|
|
|
}
|
2010-08-15 03:25:51 +02:00
|
|
|
|
|
|
|
public static <T> Enumeration<T> enumeration(Collection<T> c) {
|
|
|
|
return new IteratorEnumeration<T> (c.iterator());
|
|
|
|
}
|
2009-07-25 15:41:43 -06:00
|
|
|
|
2010-08-15 14:01:33 +02:00
|
|
|
public static <T> Comparator<T> reverseOrder(Comparator<T> cmp) {
|
|
|
|
return new ReverseComparator<T>(cmp);
|
|
|
|
}
|
|
|
|
|
2007-07-21 21:47:29 -06:00
|
|
|
static class IteratorEnumeration<T> implements Enumeration<T> {
|
|
|
|
private final Iterator<T> it;
|
|
|
|
|
|
|
|
public IteratorEnumeration(Iterator<T> it) {
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T nextElement() {
|
|
|
|
return it.next();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasMoreElements() {
|
|
|
|
return it.hasNext();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static class SynchronizedCollection<T> implements Collection<T> {
|
2007-09-26 09:19:21 -06:00
|
|
|
protected final Object lock;
|
|
|
|
protected final Collection<T> collection;
|
2007-07-21 21:47:29 -06:00
|
|
|
|
|
|
|
public SynchronizedCollection(Object lock, Collection<T> collection) {
|
|
|
|
this.lock = lock;
|
|
|
|
this.collection = collection;
|
|
|
|
}
|
|
|
|
|
2008-01-28 08:10:23 -07:00
|
|
|
public int size() {
|
2007-07-21 21:47:29 -06:00
|
|
|
synchronized (lock) { return collection.size(); }
|
|
|
|
}
|
|
|
|
|
2007-11-07 09:48:09 -07:00
|
|
|
public boolean isEmpty() {
|
|
|
|
return size() == 0;
|
|
|
|
}
|
|
|
|
|
2009-04-22 15:24:26 -06:00
|
|
|
public boolean contains(Object e) {
|
2007-07-22 13:06:21 -06:00
|
|
|
synchronized (lock) { return collection.contains(e); }
|
|
|
|
}
|
|
|
|
|
2008-01-28 08:10:23 -07:00
|
|
|
public boolean add(T e) {
|
2007-07-21 21:47:29 -06:00
|
|
|
synchronized (lock) { return collection.add(e); }
|
|
|
|
}
|
|
|
|
|
2008-02-28 11:37:10 -07:00
|
|
|
public boolean addAll(Collection<? extends T> collection) {
|
|
|
|
synchronized (lock) { return this.collection.addAll(collection); }
|
|
|
|
}
|
|
|
|
|
2009-04-22 15:24:26 -06:00
|
|
|
public boolean remove(Object e) {
|
|
|
|
synchronized (lock) { return collection.remove((T)e); }
|
2007-07-21 21:47:29 -06:00
|
|
|
}
|
|
|
|
|
2009-08-04 17:36:25 -06:00
|
|
|
public Object[] toArray() {
|
|
|
|
return toArray(new Object[size()]);
|
|
|
|
}
|
|
|
|
|
2008-02-28 11:37:10 -07:00
|
|
|
public <T> T[] toArray(T[] array) {
|
|
|
|
synchronized (lock) { return collection.toArray(array); }
|
|
|
|
}
|
|
|
|
|
2008-01-28 08:10:23 -07:00
|
|
|
public void clear() {
|
2007-07-21 21:47:29 -06:00
|
|
|
synchronized (lock) { collection.clear(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public Iterator<T> iterator() {
|
|
|
|
return new SynchronizedIterator(lock, collection.iterator());
|
|
|
|
}
|
|
|
|
}
|
2009-02-16 17:52:27 -07:00
|
|
|
|
|
|
|
static class SynchronizedMap<K,V> implements Map<K,V> {
|
|
|
|
protected final Object lock;
|
|
|
|
protected final Map<K,V> map;
|
|
|
|
|
|
|
|
SynchronizedMap(Map<K,V> map) {
|
|
|
|
this.map = map;
|
|
|
|
this.lock = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
SynchronizedMap(Object lock, Map<K,V> map) {
|
|
|
|
this.lock = lock;
|
|
|
|
this.map = map;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
synchronized (lock) { map.clear(); }
|
|
|
|
}
|
2009-04-17 08:57:49 -06:00
|
|
|
public boolean containsKey(Object key) {
|
2009-02-16 17:52:27 -07:00
|
|
|
synchronized (lock) { return map.containsKey(key); }
|
|
|
|
}
|
2009-04-17 08:57:49 -06:00
|
|
|
public boolean containsValue(Object value) {
|
2009-02-16 17:52:27 -07:00
|
|
|
synchronized (lock) { return map.containsValue(value); }
|
|
|
|
}
|
|
|
|
public Set<java.util.Map.Entry<K, V>> entrySet() {
|
|
|
|
synchronized (lock) { return new SynchronizedSet<java.util.Map.Entry<K, V>>(lock, map.entrySet()); }
|
|
|
|
}
|
2009-04-22 15:24:26 -06:00
|
|
|
public V get(Object key) {
|
2009-02-16 17:52:27 -07:00
|
|
|
synchronized (lock) { return map.get(key); }
|
|
|
|
}
|
|
|
|
public boolean isEmpty() {
|
|
|
|
synchronized (lock) { return map.isEmpty(); }
|
|
|
|
}
|
|
|
|
public Set<K> keySet() {
|
|
|
|
synchronized (lock) { return new SynchronizedSet<K>(lock, map.keySet()); }
|
|
|
|
}
|
|
|
|
public V put(K key, V value) {
|
|
|
|
synchronized (lock) { return map.put(key, value); }
|
|
|
|
}
|
|
|
|
public void putAll(Map<? extends K, ? extends V> elts) {
|
|
|
|
synchronized (lock) { map.putAll(elts); }
|
|
|
|
}
|
2009-04-22 15:24:26 -06:00
|
|
|
public V remove(Object key) {
|
2009-02-16 17:52:27 -07:00
|
|
|
synchronized (lock) { return map.remove(key); }
|
|
|
|
}
|
|
|
|
public int size() {
|
|
|
|
synchronized (lock) { return map.size(); }
|
|
|
|
}
|
|
|
|
public Collection<V> values() {
|
|
|
|
synchronized (lock) { return new SynchronizedCollection<V>(lock, map.values()); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static <K,V> Map<K,V> synchronizedMap(Map<K,V> map) {
|
|
|
|
return new SynchronizedMap<K, V> (map);
|
|
|
|
}
|
|
|
|
|
2007-07-21 21:47:29 -06:00
|
|
|
static class SynchronizedSet<T>
|
|
|
|
extends SynchronizedCollection<T>
|
|
|
|
implements Set<T>
|
|
|
|
{
|
|
|
|
public SynchronizedSet(Object lock, Set<T> set) {
|
|
|
|
super(lock, set);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static class SynchronizedIterator<T> implements Iterator<T> {
|
|
|
|
private final Object lock;
|
|
|
|
private final Iterator<T> it;
|
|
|
|
|
|
|
|
public SynchronizedIterator(Object lock, Iterator<T> it) {
|
|
|
|
this.lock = lock;
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T next() {
|
|
|
|
synchronized (lock) { return it.next(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
synchronized (lock) { return it.hasNext(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
synchronized (lock) { it.remove(); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-26 09:48:59 -06:00
|
|
|
static class ArrayListIterator<T> implements ListIterator<T> {
|
2007-07-21 21:47:29 -06:00
|
|
|
private final List<T> list;
|
2009-08-04 17:24:29 -06:00
|
|
|
private int toRemove = -1;
|
2007-09-26 09:48:59 -06:00
|
|
|
private int index;
|
2007-07-21 21:47:29 -06:00
|
|
|
|
|
|
|
public ArrayListIterator(List<T> list) {
|
2007-09-26 09:48:59 -06:00
|
|
|
this(list, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public ArrayListIterator(List<T> list, int index) {
|
2007-07-21 21:47:29 -06:00
|
|
|
this.list = list;
|
2007-09-26 09:48:59 -06:00
|
|
|
this.index = index - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasPrevious() {
|
|
|
|
return index >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T previous() {
|
|
|
|
if (hasPrevious()) {
|
2009-08-04 17:24:29 -06:00
|
|
|
toRemove = index;
|
2007-09-26 09:48:59 -06:00
|
|
|
return list.get(index--);
|
|
|
|
} else {
|
|
|
|
throw new NoSuchElementException();
|
|
|
|
}
|
2007-07-21 21:47:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public T next() {
|
|
|
|
if (hasNext()) {
|
2009-08-04 17:24:29 -06:00
|
|
|
toRemove = ++index;
|
|
|
|
return list.get(index);
|
2007-07-21 21:47:29 -06:00
|
|
|
} else {
|
|
|
|
throw new NoSuchElementException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return index + 1 < list.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
2009-08-04 17:24:29 -06:00
|
|
|
if (toRemove != -1) {
|
|
|
|
list.remove(toRemove);
|
|
|
|
index = toRemove - 1;
|
|
|
|
toRemove = -1;
|
2007-07-21 21:47:29 -06:00
|
|
|
} else {
|
|
|
|
throw new IllegalStateException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-07-03 09:16:32 -06:00
|
|
|
|
|
|
|
static class UnmodifiableSet<T> implements Set<T> {
|
2008-07-03 10:49:08 -06:00
|
|
|
Set<T> inner;
|
|
|
|
|
|
|
|
UnmodifiableSet(Set<T> inner) {
|
|
|
|
this.inner = inner;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean add(T element) {
|
|
|
|
throw new UnsupportedOperationException("not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean addAll(Collection<? extends T> collection) {
|
|
|
|
throw new UnsupportedOperationException("not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
throw new UnsupportedOperationException("not supported");
|
|
|
|
}
|
|
|
|
|
2009-04-22 15:24:26 -06:00
|
|
|
public boolean contains(Object element) {
|
2008-07-03 10:49:08 -06:00
|
|
|
return inner.contains(element);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isEmpty() {
|
|
|
|
return inner.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Iterator<T> iterator() {
|
|
|
|
return inner.iterator();
|
|
|
|
}
|
|
|
|
|
2009-04-22 15:24:26 -06:00
|
|
|
public boolean remove(Object element) {
|
2008-07-03 10:49:08 -06:00
|
|
|
throw new UnsupportedOperationException("not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
public int size() {
|
|
|
|
return inner.size();
|
|
|
|
}
|
|
|
|
|
2009-08-04 17:36:25 -06:00
|
|
|
public Object[] toArray() {
|
|
|
|
return toArray(new Object[size()]);
|
|
|
|
}
|
|
|
|
|
2008-07-03 10:49:08 -06:00
|
|
|
public <S> S[] toArray(S[] array) {
|
|
|
|
return inner.toArray(array);
|
2009-08-04 17:36:25 -06:00
|
|
|
}
|
2008-07-03 09:16:32 -06:00
|
|
|
}
|
|
|
|
|
2008-07-03 10:49:08 -06:00
|
|
|
public static <T> Set<T> unmodifiableSet(Set<T> hs) {
|
|
|
|
return new UnmodifiableSet<T>(hs);
|
|
|
|
}
|
2009-07-25 15:41:43 -06:00
|
|
|
|
|
|
|
static class KeyIterator<K, V> implements Iterator<K> {
|
|
|
|
private final Iterator<Map.Entry<K, V>> it;
|
|
|
|
|
|
|
|
public KeyIterator(Iterator<Map.Entry<K, V>> it) {
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public K next() {
|
|
|
|
return it.next().getKey();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return it.hasNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static class ValueIterator<K, V> implements Iterator<V> {
|
|
|
|
private final Iterator<Map.Entry<K, V>> it;
|
|
|
|
|
|
|
|
public ValueIterator(Iterator<Map.Entry<K, V>> it) {
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public V next() {
|
|
|
|
return it.next().getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return it.hasNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
2010-08-15 14:01:33 +02:00
|
|
|
|
|
|
|
private static final class ReverseComparator<T> implements Comparator<T> {
|
|
|
|
|
|
|
|
Comparator<T> cmp;
|
|
|
|
|
|
|
|
public ReverseComparator(Comparator<T> cmp) {
|
|
|
|
this.cmp = cmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int compare(T o1, T o2) {
|
|
|
|
return - cmp.compare(o1, o2);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2007-07-21 21:47:29 -06:00
|
|
|
}
|