/*
Реализация среды исполнения языка программирования
Объектно-ориентированный продвинутый векторный транслятор
Copyright © 2021, 2024 Малик Разработчик
Это свободная программа: вы можете перераспространять ее и/или изменять
ее на условиях Меньшей Стандартной общественной лицензии GNU в том виде,
в каком она была опубликована Фондом свободного программного обеспечения;
либо версии 3 лицензии, либо (по вашему выбору) любой более поздней версии.
Эта программа распространяется в надежде, что она будет полезной,
но БЕЗО ВСЯКИХ ГАРАНТИЙ; даже без неявной гарантии ТОВАРНОГО ВИДА
или ПРИГОДНОСТИ ДЛЯ ОПРЕДЕЛЕННЫХ ЦЕЛЕЙ. Подробнее см. в Меньшей Стандартной
общественной лицензии GNU.
Вы должны были получить копию Меньшей Стандартной общественной лицензии GNU
вместе с этой программой. Если это не так, см.
<https://www.gnu.org/licenses/>.
*/
package avt.lang.table;
public final class KeyValueTable(Object)
{
private long fldLength;
private KeyValueEntry[] fldTable;
public () {
fldTable = new KeyValueEntry[0x0fi];
}
public boolean contains(Object key) {
if(key == null) return false;
for(long2 hash = key.hashCodeAsLong2(), KeyValueEntry[] table = fldTable, KeyValueEntry entry = table[Entry.getIndex(hash, table.length)]; entry != null; entry = entry.next)
{
if(hash == entry.hash && key.equals(entry.key)) return true;
}
return false;
}
public Enumeration enumerate() {
KeyValueEntry[] table = fldTable;
int index = Array.indexOfNon(null, table, 0, 0);
return index < 0 ? null : new KeyValueEnumeration(table, index, table[index]);
}
public long length { read = fldLength }
public void operator []=(Object key, Object value) {
if(key == null)
{
throw new NullPointerException(String.format(avt.lang.package.getResourceString("null-pointer.argument"), new Object[] { "key" }));
}
if(value == null)
{
remove(key);
return;
}
put(key, value);
}
public Object operator [](Object key) {
if(key == null)
{
throw new NullPointerException(String.format(avt.lang.package.getResourceString("null-pointer.argument"), new Object[] { "key" }));
}
for(long2 hash = key.hashCodeAsLong2(), KeyValueEntry[] table = fldTable, KeyValueEntry entry = table[Entry.getIndex(hash, table.length)]; entry != null; entry = entry.next)
{
if(hash == entry.hash && key.equals(entry.key)) return entry.value;
}
return null;
}
private void rehash() {
KeyValueEntry[] oldTable = fldTable;
int oldCapacity = oldTable.length;
int newCapacity = oldCapacity << 1 | 1;
if(newCapacity < 0) newCapacity = Int.MAX_VALUE;
KeyValueEntry[] newTable = fldTable = new KeyValueEntry[newCapacity];
for(int oldIndex = oldCapacity; oldIndex-- > 0; ) for(KeyValueEntry oldEntry = oldTable[oldIndex]; oldEntry != null; )
{
int newIndex = Entry.getIndex(oldEntry.hash, newCapacity);
KeyValueEntry newEntry = oldEntry;
oldEntry = oldEntry.next;
newEntry.next = newTable[newIndex];
newTable[newIndex] = newEntry;
}
}
private void remove(Object key) {
long2 hash = key.hashCodeAsLong2();
for(KeyValueEntry[] table = fldTable, int index = Entry.getIndex(hash, table.length), KeyValueEntry prev = null, KeyValueEntry entry = table[index]; entry != null; entry = (prev = entry).next)
{
if(hash == entry.hash && key.equals(entry.key))
{
if(prev != null)
{
prev.next = entry.next;
} else
{
table[index] = entry.next;
}
fldLength--;
break;
}
}
}
private void put(Object key, Object value) {
KeyValueEntry[] table = fldTable;
long2 hash = key.hashCodeAsLong2();
long length = fldLength;
int capacity = table.length;
int index = Entry.getIndex(hash, capacity);
if(length++ >= capacity && length <= Int.MAX_VALUE)
{
rehash();
index = Entry.getIndex(hash, (table = fldTable).length);
}
table[index] = new KeyValueEntry(hash, key, value, table[index]);
fldLength = length;
}
}
class KeyValueEntry(Entry)
{
private KeyValueEntry fldNext;
private final Object fldKey;
(long2 hash, Object key, Object value, KeyValueEntry next): super(hash, value) {
fldNext = next;
fldKey = key;
}
public Object key { read = fldKey }
public KeyValueEntry next { read = fldNext, write = fldNext }
}
class KeyValueEnumeration(Enumeration)
{
private int fldIndex;
private KeyValueEntry fldEntry;
private final KeyValueEntry[] fldTable;
(KeyValueEntry[] table, int index, KeyValueEntry entry): super(entry.hash, entry.key, entry.value) {
fldIndex = index;
fldEntry = entry;
fldTable = table;
}
public boolean findNext() {
KeyValueEntry entry = fldEntry;
label0: if(entry != null)
{
if((entry = entry.next) == null)
{
int index = fldIndex;
KeyValueEntry[] table = fldTable;
if((index = Array.indexOfNon(null, table, index + 1, 0)) < 0)
{
fldEntry = null;
break label0;
}
entry = table[fldIndex = index];
}
fldEntry = entry;
setProperties(entry.hash, entry.key, entry.value);
return true;
}
return false;
}
}