From d7660799be9c9c39c90bce3b4f0fbc4fcd4cb802 Mon Sep 17 00:00:00 2001 From: Gabriel Tofvesson Date: Mon, 31 Oct 2016 22:12:39 +0100 Subject: [PATCH] Removed ShiftingMap (project for another day) Removed Shifting Set Removed Shifting Entry Added Shifting List inheriting most of it's functionality from the Shifting Set Optimized load balancing algorithm to ensure that array isn't unnecessarily resized Optimized load balancing algorithm by simplifying load calculation Added new List Iterator in addition to regular Iterator --- .../tofvesson/collections/ShiftingEntry.java | 35 -- .../tofvesson/collections/ShiftingList.java | 354 ++++++++++++++++++ .../tofvesson/collections/ShiftingMap.java | 112 ------ .../tofvesson/collections/ShiftingSet.java | 217 ----------- 4 files changed, 354 insertions(+), 364 deletions(-) delete mode 100644 src/com/tofvesson/collections/ShiftingEntry.java create mode 100644 src/com/tofvesson/collections/ShiftingList.java delete mode 100644 src/com/tofvesson/collections/ShiftingMap.java delete mode 100644 src/com/tofvesson/collections/ShiftingSet.java 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