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

ch.ethz.globis.phtree.v16.NodeIteratorListReuse Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2011-2016 ETH Zurich. All Rights Reserved.
 * Copyright 2016-2018 Tilmann Zäschke. All Rights Reserved.
 * Copyright 2019 Improbable Worlds Limited. All rights reserved.
 *
 * This file is part of the PH-Tree project.
 *
 * 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.
 */
package ch.ethz.globis.phtree.v16;

import java.util.List;

import ch.ethz.globis.phtree.PhEntry;
import ch.ethz.globis.phtree.util.Refs;
import ch.ethz.globis.phtree.v16.Node.BSTEntry;
import ch.ethz.globis.phtree.v16.bst.BSTIteratorMask;

/**
 * A NodeIterator that returns a list instead of an Iterator AND reuses the NodeIterator.
 * 
 * This seems to be slower than the simple NodeIteratorList. Why? 
 * The reason could be that the simple NIL creates many objects, but they end up in the stack
 * memory because they can not escape.
 * The NILReuse creates much less objects, but they end up in an array for reuse, which means
 * they may not be on the stack.... 
 * 
 * @author ztilmann
 *
 * @param  value type
 * @param  result type
 */
public class NodeIteratorListReuse {
	
	private class PhIteratorStack {
		@SuppressWarnings("unchecked")
		private final NodeIterator[] stack =
				Refs.newArray(NodeIteratorListReuse.NodeIterator.class, 64);
		private int size = 0;


		NodeIterator prepare() {
			NodeIterator ni = stack[size++];
			if (ni == null)  {
				ni = new NodeIterator();
				stack[size-1] = ni;
			}
			return ni;
		}

		NodeIterator pop() {
			return stack[--size];
		}
	}

	
	private final PhResultList results;
	private int maxResults;
	private long[] rangeMin;
	private long[] rangeMax;

	private final PhIteratorStack pool;
	
	private final class NodeIterator {
	
		private final BSTIteratorMask niIterator = new BSTIteratorMask();
		private long maskLower;
		private long maskUpper;


		void reinitAndRun(Node node, long[] prefix) {
			calcLimits(node, rangeMin, rangeMax, prefix);

			this.niIterator.reset(node.getRoot(), maskLower,  maskUpper, node.getEntryCount());

			getAll();
		}

		
		private void checkAndRunSubnode(Node sub, long[] subPrefix) {
			if (results.phIsPrefixValid(subPrefix, sub.getPostLen()+1)) {
				run(sub, subPrefix);
			}
		}


		@SuppressWarnings("unchecked")
		private void readValue(BSTEntry candidate) {
			//TODO avoid getting/assigning element? -> Most entries fail!
			PhEntry result = results.phGetTempEntry();
			result.setKeyInternal(candidate.getKdKey());
			result.setValueInternal((T) candidate.getValue());
			results.phOffer(result);
		}
		
		private void checkEntry(BSTEntry be) {
			Object v = be.getValue();
			if (v instanceof Node) {
				checkAndRunSubnode((Node) v, be.getKdKey());
			} else if (v != null) { 
				readValue(be);
			}
		}

		private void getAll() {
			niAllNext();
		}


		private void niAllNext() {
			niAllNextIterator();
		}
		
		private void niAllNextIterator() {
			while (niIterator.hasNextEntry() && results.size() < maxResults) {
				BSTEntry be = niIterator.nextEntry();
				checkEntry(be);
			}
		}		
		

		private void calcLimits(Node node, long[] rangeMin, long[] rangeMax, long[] prefix) {
			//create limits for the local node. there is a lower and an upper limit. Each limit
			//consists of a series of DIM bit, one for each dimension.
			//For the lower limit, a '1' indicates that the 'lower' half of this dimension does 
			//not need to be queried.
			//For the upper limit, a '0' indicates that the 'higher' half does not need to be 
			//queried.
			//
			//              ||  lowerLimit=0 || lowerLimit=1 || upperLimit = 0 || upperLimit = 1
			// =============||===================================================================
			// query lower  ||     YES             NO
			// ============ || ==================================================================
			// query higher ||                                     NO               YES
			//
			long maskHcBit = 1L << node.getPostLen();
			long maskVT = (-1L) << node.getPostLen();
			long lowerLimit = 0;
			long upperLimit = 0;
			//to prevent problems with signed long when using 64 bit
			if (maskHcBit >= 0) { //i.e. postLen < 63
				for (int i = 0; i < rangeMin.length; i++) {
					lowerLimit <<= 1;
					upperLimit <<= 1;
					long nodeBisection = (prefix[i] | maskHcBit) & maskVT; 
					if (rangeMin[i] >= nodeBisection) {
						//==> set to 1 if lower value should not be queried 
						lowerLimit |= 1L;
					}
					if (rangeMax[i] >= nodeBisection) {
						//Leave 0 if higher value should not be queried.
						upperLimit |= 1L;
					}
				}
			} else {
				//special treatment for signed longs
				//The problem (difference) here is that a '1' at the leading bit does indicate a
				//LOWER value, opposed to indicating a HIGHER value as in the remaining 63 bits.
				//The hypercube assumes that a leading '0' indicates a lower value.
				//Solution: We leave HC as it is.

				for (int i = 0; i < rangeMin.length; i++) {
					lowerLimit <<= 1;
					upperLimit <<= 1;
					if (rangeMin[i] < 0) {
						//If minimum is positive, we don't need the search negative values 
						//==> set upperLimit to 0, prevent searching values starting with '1'.
						upperLimit |= 1L;
					}
					if (rangeMax[i] < 0) {
						//Leave 0 if higher value should not be queried
						//If maximum is negative, we do not need to search positive values 
						//(starting with '0').
						//--> lowerLimit = '1'
						lowerLimit |= 1L;
					}
				}
			}
			this.maskLower = lowerLimit;
			this.maskUpper = upperLimit;
		}
	}
	
	NodeIteratorListReuse(PhResultList results) {
		this.results = results;
		this.pool = new PhIteratorStack();
	}

	List resetAndRun(Node node, long[] rangeMin, long[] rangeMax, int maxResults) {
		results.clear();
		this.rangeMin = rangeMin;
		this.rangeMax = rangeMax;
		this.maxResults = maxResults;
		run(node, null);
		return results;
	}
	
	private void run(Node node, long[] prefix) {
		NodeIterator nIt = pool.prepare();
		nIt.reinitAndRun(node, prefix);
		pool.pop();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy