2013-07-02 20:52:38 -06:00
|
|
|
/* Copyright (c) 2008-2013, 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-09-27 16:23:05 -06:00
|
|
|
package java.util;
|
|
|
|
|
2009-06-04 17:59:34 -06:00
|
|
|
import avian.PersistentSet;
|
|
|
|
import avian.Cell;
|
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
public class TreeSet<T> extends AbstractSet<T> implements Collection<T> {
|
2007-09-28 11:01:57 -06:00
|
|
|
private PersistentSet<Cell<T>> set;
|
2007-09-27 16:23:05 -06:00
|
|
|
|
2007-09-28 11:01:57 -06:00
|
|
|
public TreeSet(final Comparator<T> comparator) {
|
|
|
|
set = new PersistentSet(new Comparator<Cell<T>>() {
|
|
|
|
public int compare(Cell<T> a, Cell<T> b) {
|
|
|
|
return comparator.compare(a.value, b.value);
|
|
|
|
}
|
|
|
|
});
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
|
2008-07-14 08:45:51 -06:00
|
|
|
public TreeSet() {
|
2008-07-17 15:46:54 -06:00
|
|
|
this(new Comparator<T>() {
|
|
|
|
public int compare(T a, T b) {
|
|
|
|
return ((Comparable) a).compareTo(b);
|
|
|
|
}
|
2008-07-14 08:45:51 -06:00
|
|
|
});
|
|
|
|
}
|
2011-01-20 09:39:16 -07:00
|
|
|
|
|
|
|
public TreeSet(Collection<? extends T> collection) {
|
|
|
|
this();
|
|
|
|
|
|
|
|
for (T item: collection) {
|
|
|
|
add(item);
|
|
|
|
}
|
|
|
|
}
|
2011-09-29 18:25:41 -06:00
|
|
|
|
|
|
|
public T first() {
|
|
|
|
if (isEmpty()) throw new NoSuchElementException();
|
|
|
|
|
|
|
|
return set.first().value().value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T last() {
|
|
|
|
if (isEmpty()) throw new NoSuchElementException();
|
|
|
|
|
|
|
|
return set.last().value().value;
|
|
|
|
}
|
2008-07-14 08:45:51 -06:00
|
|
|
|
2007-09-27 16:23:05 -06:00
|
|
|
public Iterator<T> iterator() {
|
2007-09-28 11:01:57 -06:00
|
|
|
return new MyIterator<T>(set.first());
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
|
2013-11-01 20:20:05 -05:00
|
|
|
public Iterator<T> descendingIterator() {
|
|
|
|
ArrayList<T> iterable = new ArrayList<T>(this);
|
|
|
|
Collections.reverse(iterable);
|
|
|
|
return iterable.iterator();
|
|
|
|
}
|
|
|
|
|
2007-10-16 19:17:37 -06:00
|
|
|
public String toString() {
|
|
|
|
return Collections.toString(this);
|
|
|
|
}
|
|
|
|
|
2007-09-28 11:01:57 -06:00
|
|
|
public boolean add(T value) {
|
|
|
|
PersistentSet.Path<Cell<T>> p = set.find(new Cell(value, null));
|
|
|
|
if (p.fresh()) {
|
|
|
|
set = p.add();
|
|
|
|
return true;
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
2007-09-28 11:01:57 -06:00
|
|
|
return false;
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
T addAndReplace(T value) {
|
2007-09-28 11:01:57 -06:00
|
|
|
PersistentSet.Path<Cell<T>> p = set.find(new Cell(value, null));
|
|
|
|
if (p.fresh()) {
|
|
|
|
set = p.add();
|
2009-07-25 15:41:43 -06:00
|
|
|
return null;
|
2007-09-27 16:23:05 -06:00
|
|
|
} else {
|
2009-07-25 15:41:43 -06:00
|
|
|
T old = p.value().value;
|
2007-09-28 11:01:57 -06:00
|
|
|
set = p.replaceWith(new Cell(value, null));
|
2009-07-25 15:41:43 -06:00
|
|
|
return old;
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
}
|
2007-09-28 11:01:57 -06:00
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
T find(T value) {
|
|
|
|
PersistentSet.Path<Cell<T>> p = set.find(new Cell(value, null));
|
|
|
|
return p.fresh() ? null : p.value().value;
|
|
|
|
}
|
|
|
|
|
|
|
|
T removeAndReturn(T value) {
|
|
|
|
Cell<T> cell = removeCell(value);
|
|
|
|
return cell == null ? null : cell.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Cell<T> removeCell(Object value) {
|
2007-09-28 11:01:57 -06:00
|
|
|
PersistentSet.Path<Cell<T>> p = set.find(new Cell(value, null));
|
|
|
|
if (p.fresh()) {
|
2009-07-25 15:41:43 -06:00
|
|
|
return null;
|
2007-09-28 11:01:57 -06:00
|
|
|
} else {
|
2009-07-25 15:41:43 -06:00
|
|
|
Cell<T> old = p.value();
|
|
|
|
|
2007-09-28 11:01:57 -06:00
|
|
|
if (p.value().next != null) {
|
|
|
|
set = p.replaceWith(p.value().next);
|
|
|
|
} else {
|
|
|
|
set = p.remove();
|
|
|
|
}
|
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
return old;
|
2007-09-28 11:01:57 -06:00
|
|
|
}
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
|
2009-07-25 15:41:43 -06:00
|
|
|
public boolean remove(Object value) {
|
|
|
|
return removeCell(value) != null;
|
|
|
|
}
|
|
|
|
|
2007-09-27 16:23:05 -06:00
|
|
|
public int size() {
|
2012-06-13 08:41:15 -06:00
|
|
|
return set.size();
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
|
|
|
|
2007-11-07 09:48:09 -07:00
|
|
|
public boolean isEmpty() {
|
2012-07-13 15:39:57 -06:00
|
|
|
return set.size() == 0;
|
2007-11-07 09:48:09 -07:00
|
|
|
}
|
|
|
|
|
2009-04-22 15:24:26 -06:00
|
|
|
public boolean contains(Object value) {
|
2007-10-16 12:14:03 -06:00
|
|
|
return !set.find(new Cell(value, null)).fresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
set = new PersistentSet(set.comparator());
|
|
|
|
}
|
|
|
|
|
2007-09-28 11:01:57 -06:00
|
|
|
private class MyIterator<T> implements java.util.Iterator<T> {
|
|
|
|
private PersistentSet.Path<Cell<T>> path;
|
|
|
|
private PersistentSet.Path<Cell<T>> nextPath;
|
|
|
|
private Cell<T> cell;
|
|
|
|
private Cell<T> prevCell;
|
|
|
|
private Cell<T> prevPrevCell;
|
|
|
|
private boolean canRemove = false;
|
|
|
|
|
|
|
|
private MyIterator(PersistentSet.Path<Cell<T>> path) {
|
|
|
|
this.path = path;
|
|
|
|
if (path != null) {
|
|
|
|
cell = path.value();
|
|
|
|
nextPath = path.successor();
|
|
|
|
}
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|
2007-09-28 11:01:57 -06:00
|
|
|
|
|
|
|
private MyIterator(MyIterator<T> start) {
|
|
|
|
path = start.path;
|
|
|
|
nextPath = start.nextPath;
|
|
|
|
cell = start.cell;
|
|
|
|
prevCell = start.prevCell;
|
|
|
|
prevPrevCell = start.prevPrevCell;
|
|
|
|
canRemove = start.canRemove;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return cell != null || nextPath != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public T next() {
|
|
|
|
if (cell == null) {
|
|
|
|
path = nextPath;
|
|
|
|
nextPath = path.successor();
|
|
|
|
cell = path.value();
|
|
|
|
}
|
|
|
|
prevPrevCell = prevCell;
|
|
|
|
prevCell = cell;
|
|
|
|
cell = cell.next;
|
|
|
|
canRemove = true;
|
|
|
|
return prevCell.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
if (! canRemove) throw new IllegalStateException();
|
|
|
|
|
|
|
|
if (prevPrevCell != null && prevPrevCell.next == prevCell) {
|
|
|
|
// cell to remove is not the first in the list.
|
|
|
|
prevPrevCell.next = prevCell.next;
|
|
|
|
prevCell = prevPrevCell;
|
|
|
|
} else if (prevCell.next == cell && cell != null) {
|
|
|
|
// cell to remove is the first in the list, but not the last.
|
|
|
|
set = (PersistentSet) path.replaceWith(cell);
|
|
|
|
prevCell = null;
|
|
|
|
} else {
|
|
|
|
// cell is alone in the list.
|
|
|
|
set = (PersistentSet) path.remove();
|
|
|
|
path = path.successor();
|
|
|
|
if (path != null) {
|
|
|
|
prevCell = null;
|
|
|
|
cell = path.value();
|
|
|
|
path = (PersistentSet.Path) set.find((Cell) cell);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
canRemove = false;
|
|
|
|
}
|
|
|
|
}
|
2007-09-27 16:23:05 -06:00
|
|
|
}
|