All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.esri.core.geometry.IndexHashTable Maven / Gradle / Ivy

Go to download

The Esri Geometry API for Java enables developers to write custom applications for analysis of spatial data.

There is a newer version: 2.2.4
Show newest version
/*
 Copyright 1995-2015 Esri

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

 For additional information, contact:
 Environmental Systems Research Institute, Inc.
 Attn: Contracts Dept
 380 New York Street
 Redlands, California, USA 92373

 email: [email protected]
 */
package com.esri.core.geometry;

import java.util.Arrays;

final class IndexHashTable {
	// The hash function abstract class that user need to define to use the
	// IndexHashTable.
	public static abstract class HashFunction {
		public abstract int getHash(int element);

		public abstract boolean equal(int element1, int element2);

		public abstract int getHash(Object elementDescriptor);

		public abstract boolean equal(Object elementDescriptor, int element);
	}

	int m_random;
	AttributeStreamOfInt32 m_hashBuckets;
	int [] m_bit_filter; //this is aimed to speedup the find 
					     //operation and allows to have less buckets.
	IndexMultiList m_lists;
	HashFunction m_hash;

	// Create hash table. size is the bin count in the table. The hashFunction
	// is the function to use.
	public IndexHashTable(int size, HashFunction hashFunction) {
		m_hashBuckets = new AttributeStreamOfInt32(size, nullNode());
		m_lists = new IndexMultiList();
		m_hash = hashFunction;
		m_bit_filter = new int[(size*10+31) >> 5]; //10 times more bits than buckets
	}

	public void reserveElements(int capacity) {
		m_lists.reserveLists(Math.min(m_hashBuckets.size(), capacity));
		m_lists.reserveNodes(capacity);
	}

	// Adds new element to the hash table.
	public int addElement(int element, int hash) {
		int bit_bucket = hash % (m_bit_filter.length << 5);
		m_bit_filter[(bit_bucket >> 5)] |= (1 << (bit_bucket & 0x1F));
		int bucket = hash % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1) {
			list = m_lists.createList();
			m_hashBuckets.set(bucket, list);
		}
		int node = m_lists.addElement(list, element);
		return node;
	}
	
	public int addElement(int element) {
		int hash = m_hash.getHash(element);
		int bit_bucket = hash % (m_bit_filter.length << 5);
		m_bit_filter[(bit_bucket >> 5)] |= (1 << (bit_bucket & 0x1F));
		int bucket = hash % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1) {
			list = m_lists.createList();
			m_hashBuckets.set(bucket, list);
		}
		int node = m_lists.addElement(list, element);
		return node;
	}

	public void deleteElement(int element, int hash) {
		int bucket = hash % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1)
			throw new IllegalArgumentException();

		int ptr = m_lists.getFirst(list);
		int prev = -1;
		while (ptr != -1) {
			int e = m_lists.getElement(ptr);
			int nextptr = m_lists.getNext(ptr);
			if (e == element) {
				m_lists.deleteElement(list, prev, ptr);
				if (m_lists.getFirst(list) == -1) {
					m_lists.deleteList(list);// do not keep empty lists
					m_hashBuckets.set(bucket, -1);
				}
			} else {
				prev = ptr;
			}
			ptr = nextptr;
		}

	}
	
	// Removes element from the hash table.
	public void deleteElement(int element) {
		int hash = m_hash.getHash(element);
		int bucket = hash % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1)
			throw new IllegalArgumentException();

		int ptr = m_lists.getFirst(list);
		int prev = -1;
		while (ptr != -1) {
			int e = m_lists.getElement(ptr);
			int nextptr = m_lists.getNext(ptr);
			if (e == element) {
				m_lists.deleteElement(list, prev, ptr);
				if (m_lists.getFirst(list) == -1) {
					m_lists.deleteList(list);// do not keep empty lists
					m_hashBuckets.set(bucket, -1);
				}
			} else {
				prev = ptr;
			}
			ptr = nextptr;
		}

	}

	// Returns the first node in the hash table bucket defined by the given
	// hashValue.
	public int getFirstInBucket(int hashValue) {
		  int bit_bucket = hashValue % (m_bit_filter.length << 5);
		  if ((m_bit_filter[(bit_bucket >> 5)] & (1 << (bit_bucket & 0x1F))) == 0)
		    return -1;
		
		int bucket = hashValue % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1)
			return -1;

		return m_lists.getFirst(list);

	}

	// Returns next node in a bucket. Can be used together with GetFirstInBucket
	// only.
	public int getNextInBucket(int elementHandle) {
		return m_lists.getNext(elementHandle);
	}

	// Returns a node of the first element in the hash table, that is equal to
	// the given one.
	public int findNode(int element) {
		int hash = m_hash.getHash(element);
		int ptr = getFirstInBucket(hash);
		while (ptr != -1) {
			int e = m_lists.getElement(ptr);
			if (m_hash.equal(e, element)) {
				return ptr;
			}
			ptr = m_lists.getNext(ptr);
		}

		return -1;

	}

	// Returns a node to the first element in the hash table, that is equal to
	// the given element descriptor.
	public int findNode(Object elementDescriptor) {
		int hash = m_hash.getHash(elementDescriptor);
		int ptr = getFirstInBucket(hash);;
		while (ptr != -1) {
			int e = m_lists.getElement(ptr);
			if (m_hash.equal(elementDescriptor, e)) {
				return ptr;
			}
			ptr = m_lists.getNext(ptr);
		}

		return -1;

	}

	// Gets next equal node.
	public int getNextNode(int elementHandle) {
		int element = m_lists.getElement(elementHandle);
		int ptr = m_lists.getNext(elementHandle);
		while (ptr != -1) {
			int e = m_lists.getElement(ptr);
			if (m_hash.equal(e, element)) {
				return ptr;
			}
			ptr = m_lists.getNext(ptr);
		}

		return -1;

	}

	// Removes a node.
	public void deleteNode(int node) {
		int element = getElement(node);
		int hash = m_hash.getHash(element);
		int bucket = hash % m_hashBuckets.size();
		int list = m_hashBuckets.get(bucket);
		if (list == -1)
			throw new IllegalArgumentException();

		int ptr = m_lists.getFirst(list);
		int prev = -1;
		while (ptr != -1) {
			if (ptr == node) {
				m_lists.deleteElement(list, prev, ptr);
				if (m_lists.getFirst(list) == -1) {
					m_lists.deleteList(list);// do not keep empty lists
					m_hashBuckets.set(bucket, -1);
				}
				return;
			}
			prev = ptr;
			ptr = m_lists.getNext(ptr);
		}

		throw new IllegalArgumentException();

	}

	// Returns a value of the element stored in the given node.
	public int getElement(int elementHandle) {
		return m_lists.getElement(elementHandle);
	}

	// Returns any existing element from the hash table. Throws if the table is
	// empty.
	public int getAnyElement() {
		return m_lists.getFirstElement(m_lists.getFirstList());
	}

	// Returns a node for any existing element from the hash table or NullNode
	// if the table is empty.
	public int getAnyNode() {
		return m_lists.getFirst(m_lists.getFirstList());
	}

	public static int nullNode() {
		return -1;
	}

	// Removes all elements from the hash table.
	public void clear() {
		Arrays.fill(m_bit_filter, 0);
		m_hashBuckets = new AttributeStreamOfInt32(m_hashBuckets.size(),
				nullNode());
		m_lists.clear();

	}

	// Returns the number of elements in the hash table
	public int size() {
		return m_lists.getNodeCount();
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy