/** * Copyright 2014, Emory University * * 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 edu.emory.clir.clearnlp.collection.tree; import java.io.Serializable; import java.util.ArrayList; import java.util.List; import java.util.function.Function; import edu.emory.clir.clearnlp.collection.pair.ObjectIntPair; import edu.emory.clir.clearnlp.collection.triple.ObjectIntIntTriple; import edu.emory.clir.clearnlp.util.DSUtils; /** * @since 3.0.3 * @author Jinho D. Choi ({@code jinho.choi@emory.edu}) */ public class PrefixTree<K extends Comparable<K>,V> implements Serializable { private static final long serialVersionUID = 6471355272521434323L; private PrefixNode<K,V> n_root; public PrefixTree() { n_root = new PrefixNode<K,V>(); } public PrefixNode<K,V> getRoot() { return n_root; } public void setRoot(PrefixNode<K,V> node) { n_root = node; } /** * @param beginIndex inclusive * @param endIndex exclusive */ public <A>PrefixNode<K,V> add(A[] keys, int beginIndex, int endIndex, Function<A,K> f) { PrefixNode<K,V> next, curr = n_root; for (int i=beginIndex; i<endIndex; i++) { next = curr.get(keys[i]); if (next == null) { next = new PrefixNode<K,V>(); curr.put(f.apply(keys[i]), next); } curr = next; } return curr; } public <A>void set(A[] keys, V value, Function<A,K> f) { add(keys, 0, keys.length, f).setValue(value); } public <A>ObjectIntPair<V> get(A[] keys, int beginIndex, Function<A,K> f) { ObjectIntPair<V> p = new ObjectIntPair<>(); PrefixNode<K,V> curr = n_root; int i, len = keys.length; for (i=beginIndex; i<len; i++) { curr = curr.get(f.apply(keys[i])); if (curr == null) break; if (curr.hasValue()) p.set(curr.getValue(), i); } return p.o != null ? p : null; } public <A>PrefixNode<K,V> get(A[] keys, int beginIndex, int endIndex, Function<A,K> f) { PrefixNode<K,V> curr = n_root; for (int i=beginIndex; i<endIndex; i++) { curr = curr.get(f.apply(keys[i])); if (curr == null) return null; } return curr; } public <A>List<ObjectIntIntTriple<V>> getAll(A[] array, int beginIndex, Function<A,K> f, boolean removeSubset, boolean removeOverlap) { List<ObjectIntIntTriple<V>> list = new ArrayList<>(); int i, size = array.length; for (i=beginIndex; i<size; i++) getAllAux(array, i, f, list, removeSubset, removeOverlap); return list; } private <A>void getAllAux(A[] keys, int beginIndex, Function<A,K> f, List<ObjectIntIntTriple<V>> list, boolean removeSubset, boolean removeOverlap) { ObjectIntPair<V> v = get(keys, beginIndex, f); if (v == null) return; ObjectIntIntTriple<V> t = DSUtils.getLast(list); if (removeSubset && t != null && t.i2 >= v.i) return; if (removeOverlap && t != null && t.i2 >= beginIndex) { if (t.i2 - t.i1 < v.i - beginIndex) DSUtils.removeLast(list); else return; } list.add(new ObjectIntIntTriple<V>(v.o, beginIndex, v.i)); } }