001    /*
002     * $Id: Trie.java,v 1.3 2012/02/19 17:35:52 davep Exp $
003     *
004     * This file is part of McIDAS-V
005     *
006     * Copyright 2007-2012
007     * Space Science and Engineering Center (SSEC)
008     * University of Wisconsin - Madison
009     * 1225 W. Dayton Street, Madison, WI 53706, USA
010     * https://www.ssec.wisc.edu/mcidas
011     * 
012     * All Rights Reserved
013     * 
014     * McIDAS-V is built on Unidata's IDV and SSEC's VisAD libraries, and
015     * some McIDAS-V source code is based on IDV and VisAD source code.  
016     * 
017     * McIDAS-V is free software; you can redistribute it and/or modify
018     * it under the terms of the GNU Lesser Public License as published by
019     * the Free Software Foundation; either version 3 of the License, or
020     * (at your option) any later version.
021     * 
022     * McIDAS-V is distributed in the hope that it will be useful,
023     * but WITHOUT ANY WARRANTY; without even the implied warranty of
024     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
025     * GNU Lesser Public License for more details.
026     * 
027     * You should have received a copy of the GNU Lesser Public License
028     * along with this program.  If not, see http://www.gnu.org/licenses.
029     */
030    package edu.wisc.ssec.mcidasv.util.trie;
031    
032    import java.util.Map;
033    import java.util.SortedMap;
034    
035    /**
036     * Defines the interface for a prefix tree, an ordered tree data structure. For 
037     * more information, see <a href= "http://en.wikipedia.org/wiki/Trie">Tries</a>.
038     * 
039     * @author Roger Kapsi
040     * @author Sam Berlin
041     */
042    public interface Trie<K, V> extends SortedMap<K, V> {
043        
044        /**
045         * Returns a view of this Trie of all elements that are
046         * prefixed by the given key.
047         * <p>
048         * In a fixed-keysize Trie, this is essentially a 'get' operation.
049         * <p>
050         * For example, if the Trie contains 'Lime', 'LimeWire', 
051         * 'LimeRadio', 'Lax', 'Later', 'Lake', and 'Lovely', then
052         * a lookup of 'Lime' would return 'Lime', 'LimeRadio', and 'LimeWire'.
053         */
054        public SortedMap<K, V> getPrefixedBy(K key);
055        
056        /**
057         * Returns a view of this Trie of all elements that are
058         * prefixed by the length of the key.
059         * <p>
060         * Fixed-keysize Tries will not support this operation
061         * (because all keys will be the same length).
062         * <p>
063         * For example, if the Trie contains 'Lime', 'LimeWire', 
064         * 'LimeRadio', 'Lax', 'Later', 'Lake', and 'Lovely', then
065         * a lookup of 'LimePlastics' with a length of 4 would
066         * return 'Lime', 'LimeRadio', and 'LimeWire'.
067         */
068        public SortedMap<K, V> getPrefixedBy(K key, int length);
069        
070        /**
071         * Returns a view of this Trie of all elements that are prefixed
072         * by the key, starting at the given offset and for the given length.
073         * <p>
074         * Fixed-keysize Tries will not support this operation
075         * (because all keys are the same length).
076         * <p>
077         * For example, if the Trie contains 'Lime', 'LimeWire', 
078         * 'LimeRadio', 'Lax', 'Later', 'Lake', and 'Lovely', then
079         * a lookup of 'The Lime Plastics' with an offset of 4 and a 
080         * length of 4 would return 'Lime', 'LimeRadio', and 'LimeWire'.
081         */
082        public SortedMap<K, V> getPrefixedBy(K key, int offset, int length);
083        
084        /**
085         * Returns a view of this Trie of all elements that are prefixed
086         * by the number of bits in the given Key.
087         * <p>
088         * Fixed-keysize Tries can support this operation as a way to do
089         * lookups of partial keys.  That is, if the Trie is storing IP
090         * addresses, you can lookup all addresses that begin with
091         * '192.168' by providing the key '192.168.X.X' and a length of 16
092         * would return all addresses that begin with '192.168'.
093         */
094        public SortedMap<K, V> getPrefixedByBits(K key, int bitLength);
095        
096        /**
097         * Returns the value for the entry whose key is closest in a bitwise
098         * XOR metric to the given key.  This is NOT lexicographic closeness.
099         * For example, given the keys:<br>
100         *  D = 1000100 <br>
101         *  H = 1001000 <br> 
102         *  L = 1001100 <br>
103         * <p>
104         * If the Trie contained 'H' and 'L', a lookup of 'D' would return 'L',
105         * because the XOR distance between D & L is smaller than the XOR distance 
106         * between D & H. 
107         */
108        public V select(K key);
109        
110        /**
111         * Iterates through the Trie, starting with the entry whose bitwise
112         * value is closest in an XOR metric to the given key.  After the closest
113         * entry is found, the Trie will call select on that entry and continue
114         * calling select for each entry (traversing in order of XOR closeness,
115         * NOT lexicographically) until the cursor returns 
116         * <code>Cursor.SelectStatus.EXIT</code>.<br>
117         * The cursor can return <code>Cursor.SelectStatus.CONTINUE</code> to 
118         * continue traversing.<br>
119         * <code>Cursor.SelectStatus.REMOVE_AND_EXIT</code> is used to remove the current element
120         * and stop traversing.
121         * <p>
122         * Note: The {@link Cursor.SelectStatus#REMOVE} operation is not supported.
123         * 
124         * @return The entry the cursor returned EXIT on, or null if it continued
125         *         till the end.
126         */
127        public Map.Entry<K,V> select(K key, Cursor<? super K, ? super V> cursor);
128        
129        /**
130         * Traverses the Trie in lexicographical order. <code>Cursor.select</code> 
131         * will be called on each entry.<p>
132         * The traversal will stop when the cursor returns <code>Cursor.SelectStatus.EXIT</code>.<br>
133         * <code>Cursor.SelectStatus.CONTINUE</code> is used to continue traversing.<br>
134         * <code>Cursor.SelectStatus.REMOVE</code> is used to remove the element that was 
135         * selected and continue traversing.<br>
136         * <code>Cursor.SelectStatus.REMOVE_AND_EXIT</code> is used to remove the current element
137         * and stop traversing.
138         *   
139         * @return The entry the cursor returned EXIT on, or null if it continued
140         *         till the end.
141         */
142        public Map.Entry<K,V> traverse(Cursor<? super K, ? super V> cursor);
143        
144        /**
145         * An interface used by a {@link Trie}. A {@link Trie} selects items by 
146         * closeness and passes the items to the <code>Cursor</code>. You can then 
147         * decide what to do with the key-value pair and the return value 
148         * from {@link #select(java.util.Map.Entry)} tells the <code>Trie</code> 
149         * what to do next.
150         * <p>
151         * <code>Cursor</code> returns status/selection status might be:
152         * <table cellspace="5">
153         * <tr><td><b>Return Value</b></td><td><b>Status</b></td></tr>
154         * <tr><td>EXIT</td><td>Finish the Trie operation</td></tr>
155         * <tr><td>CONTINUE</td><td>Look at the next element in the traversal</td></tr>
156         * <tr><td>REMOVE_AND_EXIT</td><td>Remove the entry and stop iterating</td></tr>
157         * <tr><td>REMOVE</td><td>Remove the entry and continue iterating</td></tr>
158         * </table>
159    
160         * Note: {@link Trie#select(Object, org.limewire.collection.Trie.Cursor)} does
161         * not support <code>REMOVE</code>.
162         *
163         * @param <K> Key Type
164         * @param <V> Key Value
165         */
166        public static interface Cursor<K, V> {
167            
168            /**
169             * Notification that the Trie is currently looking at the given entry.
170             * Return <code>EXIT</code> to finish the Trie operation, 
171             * <code>CONTINUE</code> to look at the next entry, <code>REMOVE</code>
172             * to remove the entry and continue iterating, or
173             * <code>REMOVE_AND_EXIT</code> to remove the entry and stop iterating. 
174             * Not all operations support <code>REMOVE</code>.
175             * 
176             */
177            public SelectStatus select(Map.Entry<? extends K, ? extends V> entry);
178         
179            /** The mode during selection.      */
180            public static enum SelectStatus {
181                EXIT, CONTINUE, REMOVE, REMOVE_AND_EXIT;
182            }
183        }
184    }
185