diff --git a/src/com/tofvesson/collections/ShiftingEntry.java b/src/com/tofvesson/collections/ShiftingEntry.java deleted file mode 100644 index 1b94169..0000000 --- a/src/com/tofvesson/collections/ShiftingEntry.java +++ /dev/null @@ -1,35 +0,0 @@ -package com.tofvesson.collections; - -import java.util.Map; - -import static com.tofvesson.collections.ShiftingSet.empty; - -public class ShiftingEntry implements Map.Entry{ - - private final ShiftingSet keys; - private final ShiftingSet values; - int pos; - - public ShiftingEntry(ShiftingSet keys, ShiftingSet values, int pos){ - this.keys = keys; - this.values = values; - this.pos = pos; - } - - @Override - public K getKey() { - return keys.entries[pos]==empty?null:(K)keys.entries[pos]; - } - - @Override - public V getValue() { - return values.entries[pos]==empty?null:(V)values.entries[pos]; - } - - @Override - public V setValue(V value) { - V v = getValue(); - if(keys.entries[pos]!=null) values.entries[pos] = value; - return v; - } -} diff --git a/src/com/tofvesson/collections/ShiftingList.java b/src/com/tofvesson/collections/ShiftingList.java new file mode 100644 index 0000000..7aa2bd7 --- /dev/null +++ b/src/com/tofvesson/collections/ShiftingList.java @@ -0,0 +1,354 @@ +package com.tofvesson.collections; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +@SuppressWarnings("unchecked") +public class ShiftingList implements List { + /** + * Holder for entries. Dur to reasons, the array holds objects. + */ + Object[] entries; + + /** + * Populated entries. + */ + int pop = 0; + + /** + * Maximum size of set. + */ + final int maxSize; + + /** + * Load factor used when calculating how to resize array. + */ + double load; + + /** + * Internal reference to reserved, unpopulated entries. + */ + static final Empty empty = new Empty(); + + public ShiftingList(int maxSize, double load){ + this.maxSize = maxSize<1?20:maxSize; + this.load = load>0.99||load<0.1?0.75:load; + entries = new Object[1]; + } + + public ShiftingList(int maxSize){ + this(maxSize, 0.75); + } + + @Override + public int size() { + return pop; + } + + @Override + public boolean isEmpty() { + return pop==0; + } + + @Override + public boolean contains(Object o) { + if(o==empty) return false; + for(Object o1 : entries) if(o.equals(o1)) return true; + return false; + } + + @Override + public Object[] toArray() { + Object[] o = new Object[pop]; + System.arraycopy(entries, 0, o, 0, pop); + for(int i = 0; i T[] toArray(T[] a) { + for(int i = 0; i c) { + boolean b = true; + for(Object o : c) b &= contains(o); + return b; + } + + @Override + public boolean addAll(Collection c) { + ArrayList l = new ArrayList<>(c); + preparePopulate(c.size()); + for(int i = 0; i c) { + if(index>=entries.length || index<0 || c.size()==0) return false; + ArrayList l = new ArrayList<>(c); + for(int i = 0; imaxSize) for(int i = pop+l.size()-maxSize; i c) { + int removed = 0; + for(int i = 0; i c) { + int removed = 0; + for(int i = 0; ientries.length) adaptLoad(accountFor); // If new elements exceed limit, adapt load + if(accountFor>entries.length) return; // No need to delete elements if new values exceed limit + System.arraycopy(entries, 0, entries, accountFor, entries.length-accountFor); // Shift array elements to account for new elements + } + + @Override + public E get(int i){ + if(i>pop) return null; + return entries[i]==empty?null:(E)entries[i]; + } + + @Override + public E set(int index, E element) { + if(index > pop) return null; + E e = get(index); + entries[index] = element; + return e; + } + + @Override + public void add(int index, E element) { + if(index>=entries.length || index<0) return; + Object o = element==null?empty:element; + pop = pop==maxSize?pop:pop+1; + if(pop==entries.length) adaptLoad(); + if(index==entries.length-1){ + entries[index] = o; + return; + } + System.arraycopy(entries, index, entries, index+1, entries.length-(index+1)); + entries[index] = o; + } + + @Override + public E remove(int index) { + if(index>pop || index<0) return null; + E e = entries[index]==empty?null:(E)entries[index]; + entries[index] = null; + shift(); + if(--pop0; --i) + if(o.equals(entries[i])) + return i; + return -1; + } + + @Override + public Iterator iterator(){ + return new Iterator<>(); + } + + @Override + public ListIterator listIterator() { + return new ListIterator<>(); + } + + @Override + public ListIterator listIterator(int index) { + if(index<0) throw new RuntimeException("Invalid starting point for iterator defined: "+index); + return new ListIterator<>(index); + } + + @Override + public List subList(int fromIndex, int toIndex) { + if(fromIndex<0 || fromIndex>=toIndex || fromIndex>pop || toIndex>pop) return new ArrayList<>(); + ShiftingList l = new ShiftingList<>(maxSize); + for(int i = toIndex-1; i>fromIndex; --i) l.add(entries[i]==empty?null:(E)entries[i]); + return l; + } + + public class Iterator implements java.util.Iterator{ + + int counter = 0; + + @Override + public boolean hasNext() { + return counter implements java.util.ListIterator{ + + protected int counter = 0; + protected boolean opNxt = false; + private Object pEl = null; + + public ListIterator(){} + public ListIterator(int start){ counter = start; } + + @Override + public boolean hasNext() { + return counter0&&ShiftingList.this.pop!=0; + } + + @Override + public V previous() { + opNxt = false; + return (V)(pEl=entries[--counter]==empty?null:entries[counter]); + } + + @Override + public int nextIndex() { + return counter+1 implements Map { - - protected final ShiftingSet keys; - protected final ShiftingSet values; - protected final ShiftingSet> entries; - - public ShiftingMap(int maxSize, double load){ - keys = new ShiftingSet<>(maxSize, load); - values = new ShiftingSet<>(maxSize, load); - entries = new ShiftingSet<>(maxSize, load); - } - - public ShiftingMap(int maxSize){ - this(maxSize, 0.75); - } - - @Override - public int size() { - return keys.size(); - } - - @Override - public boolean isEmpty() { - return keys.isEmpty(); - } - - @Override - public boolean containsKey(Object key) { - return keys.contains(key); - } - - @Override - public boolean containsValue(Object value) { - return values.contains(value); - } - - @Override - public V get(Object key) { - int i = keys.indexOf(key); - return i!=-1?(V)values.entries[i]:null; - } - - @Override - public V put(K key, V value) { - V v=null; - int i = keys.indexOf(key); - if(i!=-1){ - v = values.entries[i]!= empty?(V)values.entries[i]:null; - values.entries[i] = value!=null?value:empty; - }else{ - for(Entry e : entries) ++((ShiftingEntry)e).pos; - entries.add(new ShiftingEntry<>(keys, values, 0)); - keys.add(key); - values.add(value); - } - return v; - } - - @Override - public V remove(Object key) { - V v = get(key); - if(keys.contains(key)){ - values.entries[keys.indexOf(key)] = null; - values.shift(); - values.adaptLoad(-1); - keys.remove(key); - entries.entries[entries.size()-1] = null; - entries.adaptLoad(-1); - } - return v; - } - - @Override - public void putAll(Map m) { - keys.stream().filter(m::containsKey).forEach(k -> { - put(k, m.get(k)); - m.remove(k); - }); - if(m.size()(keys, values, m.size()-i)); - keys.addAll(m.keySet()); - values.addAll(m.values()); - } - - @Override - public void clear() { - entries.clear(); - keys.clear(); - values.clear(); - } - - @Override - public Set keySet() { - return keys; - } - - @Override - public Collection values() { - return values; - } - - @Override - public Set> entrySet() { - return entries; - } -} diff --git a/src/com/tofvesson/collections/ShiftingSet.java b/src/com/tofvesson/collections/ShiftingSet.java deleted file mode 100644 index a6813f2..0000000 --- a/src/com/tofvesson/collections/ShiftingSet.java +++ /dev/null @@ -1,217 +0,0 @@ -package com.tofvesson.collections; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Set; - -public class ShiftingSet implements Set { - - /** - * Holder for entries. Dur to reasons, the array holds objects. - */ - Object[] entries; - - /** - * Populated entries. - */ - int pop = 0; - - /** - * Maximum size of set. - */ - final int maxSize; - - /** - * Load factor used when calculating how to resize array. - */ - double load; - - static final Empty empty = new Empty(); - - public ShiftingSet(int maxSize, double load){ - this.maxSize = maxSize>1?20:maxSize; - this.load = load>1||load<0.1?0.75:load; - entries = new Object[1]; - } - - public ShiftingSet(int maxSize){ - this(maxSize, 0.75); - } - - @Override - public int size() { - return pop; - } - - @Override - public boolean isEmpty() { - return pop==0; - } - - @Override - public boolean contains(Object o) { - if(o==empty) return false; - for(Object o1 : entries) if(o.equals(o1)) return true; - return false; - } - - @Override - public Iterator iterator() { - return new Iterator<>(); - } - - @Override - public Object[] toArray() { - Object[] o = new Object[pop]; - System.arraycopy(entries, 0, o, 0, pop); - return o; - } - - @Override - public T[] toArray(T[] a) { - for(int i = 0; i c) { - boolean b = true; - for(Object o : c) b &= contains(o); - return b; - } - - @Override - public boolean addAll(Collection c) { - ArrayList l = new ArrayList<>(c); - preparePopulate(c.size()); - for(int i = 0; i c) { - int removed = 0; - for(int i = 0; i c) { - int removed = 0; - for(int i = 0; ientries.length) adaptLoad(accountFor); // If new elements exceed limit, adapt load - if(accountFor>entries.length) return; // If the expected new load still exceeds the limit, no need to delete elements - System.arraycopy(entries, 0, entries, accountFor, entries.length-accountFor); // Shift array elements to account for new elements - } - - public class Iterator implements java.util.Iterator{ - - int counter = 0; - - @Override - public boolean hasNext() { - return counter