This is a java program to implement Hash Tree. In computer science, a hash tree (or hash trie) is a persistent data structure that can be used to implement sets and maps, intended to replace hash tables in purely functional programming. In its basic form, a hash tree stores the hashes of its keys, regarded as strings of bits, in a trie, with the actual keys and (optional) values stored at the trie’s “final” nodes.
Here is the source code of the Java Program to Implement Hash Tree. The Java program is successfully compiled and run on a Windows system. The program output is also shown below.
package com.maixuanviet.datastructures;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
interface HashTreeTraverser
{
public void addNode(Object node, HashTree subTree);
public void subtractNode();
public void processPath();
}
class TreeSearcher implements HashTreeTraverser
{
Object target;
HashTree result;
public TreeSearcher(Object t)
{
target = t;
}
public HashTree getResult()
{
return result;
}
public void addNode(Object node, HashTree subTree)
{
result = subTree.getTree(target);
if (result != null)
{
throw new RuntimeException("found"); // short circuit traversal
// when found
}
}
@Override
public void subtractNode()
{
}
@Override
public void processPath()
{
}
}
class ConvertToString implements HashTreeTraverser
{
StringBuffer string = new StringBuffer(getClass().getName() + "{");
StringBuffer spaces = new StringBuffer();
int depth = 0;
public void addNode(Object key, HashTree subTree)
{
depth++;
string.append("\n" + getSpaces() + key + " {");
}
public void subtractNode()
{
string.append("\n" + getSpaces() + "}");
depth--;
}
public void processPath()
{
}
public String toString()
{
string.append("\n}");
return string.toString();
}
private String getSpaces()
{
if (spaces.length() < depth * 2)
{
while (spaces.length() < depth * 2)
{
spaces.append(" ");
}
}
else if (spaces.length() > depth * 2)
{
spaces.setLength(depth * 2);
}
return spaces.toString();
}
}
@SuppressWarnings({ "rawtypes", "unchecked" })
public class HashTree implements Serializable, Map
{
private static final long serialVersionUID = 5526070397395889719L;
public HashTree()
{
data = new HashMap();
}
public HashTree(Object key)
{
data = new HashMap();
data.put(key, new HashTree());
}
public void putAll(Map map)
{
if (map instanceof HashTree)
{
this.add((HashTree) map);
}
else
{
throw new UnsupportedOperationException(
"can only putAll other HashTree objects");
}
}
public Set entrySet()
{
return data.entrySet();
}
public boolean containsValue(Object value)
{
return data.containsValue(value);
}
public Object put(Object key, Object value)
{
Object previous = data.get(key);
add(key, value);
return previous;
}
public void clear()
{
data.clear();
}
public Collection values()
{
return data.values();
}
public void add(Object key, HashTree subTree)
{
add(key);
getTree(key).add(subTree);
}
public void add(HashTree newTree)
{
Iterator iter = newTree.list().iterator();
while (iter.hasNext())
{
Object item = iter.next();
add(item);
getTree(item).add(newTree.getTree(item));
}
}
public HashTree(Collection keys)
{
data = new HashMap();
Iterator it = keys.iterator();
while (it.hasNext())
{
data.put(it.next(), new HashTree());
}
}
public HashTree(Object[] keys)
{
data = new HashMap();
for (int x = 0; x < keys.length; x++)
{
data.put(keys[x], new HashTree());
}
}
public boolean containsKey(Object o)
{
return data.containsKey(o);
}
public boolean isEmpty()
{
return data.isEmpty();
}
public void set(Object key, Object value)
{
data.put(key, createNewTree(value));
}
public void set(Object key, HashTree t)
{
data.put(key, t);
}
public void set(Object key, Object[] values)
{
data.put(key, createNewTree(Arrays.asList(values)));
}
public void set(Object key, Collection values)
{
data.put(key, createNewTree(values));
}
public void set(Object[] treePath, Object[] values)
{
if (treePath != null && values != null)
{
set(Arrays.asList(treePath), Arrays.asList(values));
}
}
public void set(Object[] treePath, Collection values)
{
if (treePath != null)
{
set(Arrays.asList(treePath), values);
}
}
public void set(Collection treePath, Object[] values)
{
HashTree tree = addTreePath(treePath);
tree.set(Arrays.asList(values));
}
public void set(Collection values)
{
clear();
this.add(values);
}
public void set(Collection treePath, Collection values)
{
HashTree tree = addTreePath(treePath);
tree.set(values);
}
public HashTree add(Object key)
{
if (!data.containsKey(key))
{
HashTree newTree = createNewTree();
data.put(key, newTree);
return newTree;
}
else
{
return getTree(key);
}
}
public void add(Object[] keys)
{
for (int x = 0; x < keys.length; x++)
{
add(keys[x]);
}
}
public void add(Collection keys)
{
Iterator it = keys.iterator();
while (it.hasNext())
{
add(it.next());
}
}
public HashTree add(Object key, Object value)
{
add(key);
return getTree(key).add(value);
}
public void add(Object key, Object[] values)
{
add(key);
getTree(key).add(values);
}
public void add(Object key, Collection values)
{
add(key);
getTree(key).add(values);
}
public void add(Object[] treePath, Object[] values)
{
if (treePath != null)
{
add(Arrays.asList(treePath), Arrays.asList(values));
}
}
public void add(Object[] treePath, Collection values)
{
if (treePath != null)
{
add(Arrays.asList(treePath), values);
}
}
public HashTree add(Object[] treePath, Object value)
{
return add(Arrays.asList(treePath), value);
}
public void add(Collection treePath, Object[] values)
{
HashTree tree = addTreePath(treePath);
tree.add(Arrays.asList(values));
}
public HashTree add(Collection treePath, Object value)
{
HashTree tree = addTreePath(treePath);
return tree.add(value);
}
public void add(Collection treePath, Collection values)
{
HashTree tree = addTreePath(treePath);
tree.add(values);
}
protected HashTree addTreePath(Collection treePath)
{
HashTree tree = this;
Iterator iter = treePath.iterator();
while (iter.hasNext())
{
Object temp = iter.next();
tree.add(temp);
tree = tree.getTree(temp);
}
return tree;
}
public HashTree getTree(Object key)
{
return (HashTree) data.get(key);
}
public Object get(Object key)
{
return getTree(key);
}
public HashTree getTree(Object[] treePath)
{
if (treePath != null)
{
return getTree(Arrays.asList(treePath));
}
else
{
return this;
}
}
public Object clone()
{
HashTree newTree = new HashTree();
cloneTree(newTree);
return newTree;
}
protected void cloneTree(HashTree newTree)
{
Iterator iter = list().iterator();
while (iter.hasNext())
{
Object key = iter.next();
newTree.set(key, (HashTree) getTree(key).clone());
}
}
protected HashTree createNewTree()
{
return new HashTree();
}
protected HashTree createNewTree(Object key)
{
return new HashTree(key);
}
protected HashTree createNewTree(Collection values)
{
return new HashTree(values);
}
public HashTree getTree(Collection treePath)
{
return getTreePath(treePath);
}
public Collection list()
{
return data.keySet();
}
public Collection list(Object key)
{
HashTree temp = (HashTree) data.get(key);
if (temp != null)
{
return temp.list();
}
else
{
return null;
}
}
public Object remove(Object key)
{
return data.remove(key);
}
public Collection list(Object[] treePath)
{
if (treePath != null)
{
return list(Arrays.asList(treePath));
}
else
{
return list();
}
}
public Collection list(Collection treePath)
{
return getTreePath(treePath).list();
}
public Object replace(Object currentKey, Object newKey)
{
HashTree tree = getTree(currentKey);
data.remove(currentKey);
data.put(newKey, tree);
return null;
}
public Object[] getArray()
{
return data.keySet().toArray();
}
public Object[] getArray(Object key)
{
return getTree(key).getArray();
}
public Object[] getArray(Object[] treePath)
{
if (treePath != null)
{
return getArray(Arrays.asList(treePath));
}
else
{
return getArray();
}
}
public Object[] getArray(Collection treePath)
{
HashTree tree = getTreePath(treePath);
return tree.getArray();
}
protected HashTree getTreePath(Collection treePath)
{
HashTree tree = this;
Iterator iter = treePath.iterator();
while (iter.hasNext())
{
Object temp = iter.next();
tree = tree.getTree(temp);
}
return tree;
}
public int hashCode()
{
return data.hashCode() * 7;
}
public boolean equals(Object o)
{
if (!(o instanceof HashTree))
return false;
HashTree oo = (HashTree) o;
if (oo.size() != this.size())
return false;
return data.equals(oo.data);
}
public Set keySet()
{
return data.keySet();
}
public HashTree search(Object key)
{
HashTree result = getTree(key);
if (result != null)
{
return result;
}
TreeSearcher searcher = new TreeSearcher(key);
try
{
traverse(searcher);
}
catch (Exception e)
{
// do nothing - means object is found
}
return searcher.getResult();
}
void readObject(ObjectInputStream ois) throws ClassNotFoundException,
IOException
{
ois.defaultReadObject();
}
void writeObject(ObjectOutputStream oos) throws IOException
{
oos.defaultWriteObject();
}
public int size()
{
return data.size();
}
public void traverse(HashTreeTraverser visitor)
{
Iterator iter = list().iterator();
while (iter.hasNext())
{
Object item = iter.next();
visitor.addNode(item, getTree(item));
getTree(item).traverseInto(visitor);
}
}
private void traverseInto(HashTreeTraverser visitor)
{
if (list().size() == 0)
{
visitor.processPath();
}
else
{
Iterator iter = list().iterator();
while (iter.hasNext())
{
Object item = iter.next();
visitor.addNode(item, getTree(item));
getTree(item).traverseInto(visitor);
}
}
visitor.subtractNode();
}
public String toString()
{
ConvertToString converter = new ConvertToString();
traverse(converter);
return converter.toString();
}
protected Map data;
public static void main(String args[])
{
Collection treePath = Arrays
.asList(new String[] { "1", "2", "3", "4" });
HashTree tree = new HashTree();
tree.add(treePath, "value");
HashTree tree1 = new HashTree("abcd");
HashTree tree2 = new HashTree("abcd");
HashTree tree3 = new HashTree("abcde");
HashTree tree4 = new HashTree("abcde");
System.out.println("Is tree1 equals tree2: " + tree1.equals(tree1));
System.out.println("Is hashcodes of tree1 and tree2 are equal: "
+ (tree1.hashCode() == tree2.hashCode()));
System.out.println("Is tree3 equals tree3: " + tree3.equals(tree3));
tree1.add("abcd", tree3);
System.out.println("Is modified tree1 is equal to tree3: "
+ tree1.equals(tree2));
tree2.add("abcd", tree4);
System.out.println("Is modified tree2 equals tree1: "
+ tree1.equals(tree2));
System.out.println("Is hashcodes are equal: "
+ (tree1.hashCode() == tree2.hashCode()));
}
}
Output:
$ javac HashTree.java $ java HashTree Is tree1 equals tree2: true Is hashcodes of tree1 and tree2 are equal: true Is tree3 equals tree3: true Is modified tree1 is equal to tree3: false Is modified tree2 equals tree1: true Is hashcodes are equal: true
Related posts:
Spring Cloud – Bootstrapping
Tránh lỗi NullPointerException trong Java như thế nào?
Java Program to Implement Shoelace Algorithm
Java Program to Implement Caesar Cypher
Java Program for Topological Sorting in Graphs
Tạo chương trình Java đầu tiên sử dụng Eclipse
How to Replace Many if Statements in Java
How to Kill a Java Thread
Java Program to Solve a Matching Problem for a Given Specific Case
Java Program to Generate All Possible Combinations Out of a, b, c, d, e
Returning Custom Status Codes from Spring Controllers
Java Program to find the maximum subarray sum O(n^2) time(naive method)
Send email with JavaMail
Java Program to Describe the Representation of Graph using Incidence Matrix
Guide to PriorityBlockingQueue in Java
Giới thiệu Swagger – Công cụ document cho RESTfull APIs
Java Program to Implement CopyOnWriteArraySet API
Java Program to Implement Segment Tree
Introduction to Spring Data REST
The Basics of Java Security
Spring Boot - File Handling
Java – Reader to InputStream
Debug a JavaMail Program
Java Program to Check whether Directed Graph is Connected using DFS
Spring Data – CrudRepository save() Method
Hướng dẫn Java Design Pattern – Command
Java Program to Implement Bloom Filter
OAuth2 for a Spring REST API – Handle the Refresh Token in AngularJS
Java Program to Implement Red Black Tree
How to Convert List to Map in Java
Template Engines for Spring
Custom HTTP Header with the HttpClient