Skip to content

Instantly share code, notes, and snippets.

@dkuppitz
Created July 12, 2016 15:18
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save dkuppitz/de1da44230c2b535f915432d8b628569 to your computer and use it in GitHub Desktop.
Save dkuppitz/de1da44230c2b535f915432d8b628569 to your computer and use it in GitHub Desktop.
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.tinkerpop.gremlin.process.traversal.step.util;
import java.io.Serializable;
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
* @author Marko A. Rodriguez (http://markorodriguez.com)
*/
public class Tree<T> extends LinkedHashSet<Map.Entry<T, Tree<T>>> implements Serializable {
private final Map<T, Tree<T>> map = new HashMap<>();
public Tree() {
super();
}
@SafeVarargs
public Tree(final T... children) {
this();
for (final T t : children) {
put(t, new Tree<>());
}
}
@SafeVarargs
public Tree(final Map.Entry<T, Tree<T>>... children) {
this();
for (final Map.Entry<T, Tree<T>> entry : children) {
this.put(entry);
}
}
public void put(final T t, final Tree<T> tree) {
map.put(t, tree);
super.add(new AbstractMap.SimpleEntry<>(t, tree));
}
private void put(final Map.Entry<T, Tree<T>> entry) {
this.put(entry.getKey(), entry.getValue());
}
public Set<T> keySet() {
return map.keySet();
}
public Collection<Tree<T>> values() {
return map.values();
}
public Set<Map.Entry<T, Tree<T>>> entrySet() {
return this;
}
public boolean containsKey(final T key) {
return map.containsKey(key);
}
public Tree<T> get(final T key) {
return map.get(key);
}
public List<Tree<T>> getTreesAtDepth(final int depth) {
List<Tree<T>> currentDepth = Collections.singletonList(this);
for (int i = 0; i < depth; i++) {
if (i == depth - 1) {
return currentDepth;
} else {
final List<Tree<T>> temp = new ArrayList<Tree<T>>();
for (final Tree<T> t : currentDepth) {
temp.addAll(t.values());
}
currentDepth = temp;
}
}
return Collections.emptyList();
}
public List<T> getObjectsAtDepth(final int depth) {
final List<T> list = new ArrayList<T>();
for (final Tree<T> t : this.getTreesAtDepth(depth)) {
list.addAll(t.keySet());
}
return list;
}
public List<Tree<T>> getLeafTrees() {
final List<Tree<T>> leaves = new ArrayList<>();
List<Tree<T>> currentDepth = Collections.singletonList(this);
boolean allLeaves = false;
while (!allLeaves) {
allLeaves = true;
final List<Tree<T>> temp = new ArrayList<>();
for (final Tree<T> t : currentDepth) {
if (t.isLeaf()) {
for (final Map.Entry<T, Tree<T>> t2 : t) {
leaves.add(new Tree<T>(t2));
}
} else {
allLeaves = false;
temp.addAll(t.values());
}
}
currentDepth = temp;
}
return leaves;
}
public List<T> getLeafObjects() {
final List<T> leaves = new ArrayList<T>();
for (final Tree<T> t : this.getLeafTrees()) {
leaves.addAll(t.keySet());
}
return leaves;
}
public boolean isLeaf() {
final Collection<Tree<T>> values = this.values();
return values.iterator().next().isEmpty();
}
public void addTree(final Tree<T> tree) {
tree.forEach(entry -> {
if (this.containsKey(entry.getKey())) {
this.get(entry.getKey()).addTree(entry.getValue());
} else {
this.put(entry);
}
});
}
public List<Tree<T>> splitParents() {
if (this.keySet().size() == 1) {
return Collections.singletonList(this);
} else {
final List<Tree<T>> parents = new ArrayList<>();
this.forEach(entry -> {
final Tree<T> parentTree = new Tree<>();
parentTree.put(entry);
parents.add(parentTree);
});
return parents;
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment