
com.questdb.griffin.PostOrderTreeTraversalAlgo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of questdb-core Show documentation
Show all versions of questdb-core Show documentation
QuestDB is High Performance Time Series Database
The newest version!
/*******************************************************************************
* ___ _ ____ ____
* / _ \ _ _ ___ ___| |_| _ \| __ )
* | | | | | | |/ _ \/ __| __| | | | _ \
* | |_| | |_| | __/\__ \ |_| |_| | |_) |
* \__\_\\__,_|\___||___/\__|____/|____/
*
* Copyright (C) 2014-2019 Appsicle
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License, version 3,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see .
*
******************************************************************************/
package com.questdb.griffin;
import com.questdb.griffin.model.ExpressionNode;
import com.questdb.std.IntStack;
import java.util.ArrayDeque;
final public class PostOrderTreeTraversalAlgo {
private final ArrayDeque stackBackup = new ArrayDeque<>();
private final IntStack indexStackBackup = new IntStack();
private final IntStack backupDepth = new IntStack();
private final ArrayDeque stack = new ArrayDeque<>();
private final IntStack indexStack = new IntStack();
public void backup() {
int size = stack.size();
backupDepth.push(size);
for (int i = 0; i < size; i++) {
stackBackup.push(stack.poll());
indexStackBackup.push(indexStack.pop());
}
}
public void restore() {
if (backupDepth.size() > 0) {
int size = backupDepth.pop();
for (int i = 0; i < size; i++) {
stack.push(stackBackup.poll());
indexStack.push(indexStackBackup.pop());
}
}
}
public void traverse(ExpressionNode node, Visitor visitor) throws SqlException {
backup();
try {
// post-order iterative tree traversal
// see http://en.wikipedia.org/wiki/Tree_traversal
stack.clear();
indexStack.clear();
ExpressionNode lastVisited = null;
while (!stack.isEmpty() || node != null) {
if (node != null) {
stack.push(node);
indexStack.push(0);
node = node.rhs;
} else {
ExpressionNode peek = stack.peek();
assert peek != null;
if (peek.paramCount < 3) {
if (peek.lhs != null && lastVisited != peek.lhs) {
node = peek.lhs;
} else {
visitor.visit(peek);
lastVisited = stack.poll();
indexStack.pop();
}
} else {
int index = indexStack.peek();
if (index < peek.paramCount) {
node = peek.args.getQuick(index);
indexStack.update(index + 1);
} else {
visitor.visit(peek);
lastVisited = stack.poll();
indexStack.pop();
}
}
}
}
} finally {
restore();
}
}
public interface Visitor {
void visit(ExpressionNode node) throws SqlException;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy