Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support moving nodes and refactor to eliminate recursion #14

Merged
merged 11 commits into from
Aug 9, 2024
238 changes: 190 additions & 48 deletions c++/cavl.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
#include <cstdint>
#include <tuple>
#include <type_traits>
#include <utility>

/// If CAVL is used in throughput-critical code, then it is recommended to disable assertion checks as they may
/// be costly in terms of execution time.
Expand Down Expand Up @@ -162,11 +163,7 @@ class Node
static auto min(const Node* const root) noexcept -> const Derived* { return extremum(root, false); }
static auto max(const Node* const root) noexcept -> const Derived* { return extremum(root, true); }

// NOLINTBEGIN(misc-no-recursion)

/// In-order or reverse-in-order traversal of the tree; the visitor is invoked with a reference to each node.
/// Required stack depth is bounded by less than 2*log2(size),
/// hence no Sonar cpp:S925 and `NOLINT(misc-no-recursion)` exceptions.
/// If the return type is non-void, then it shall be default-constructable and convertible to bool; in this case,
/// traversal will stop when the first true value is returned, which is propagated back to the caller; if none
/// of the calls returned true or the tree is empty, a default value is constructed and returned.
Expand All @@ -175,63 +172,37 @@ class Node
static auto traverse(Derived* const root, const Vis& visitor, const bool reverse = false) //
-> std::enable_if_t<!std::is_void<R>::value, R>
{
if (Node* const n = root)
serges147 marked this conversation as resolved.
Show resolved Hide resolved
{
// NOLINTNEXTLINE(*-qualified-auto)
if (auto t = Node::traverse<Vis>(down(n->lr[reverse]), visitor, reverse)) // NOSONAR cpp:S925
{
return t;
}
if (auto t = visitor(*root)) // NOLINT(*-qualified-auto)
{
return t;
}
return Node::traverse<Vis>(down(n->lr[!reverse]), visitor, reverse); // NOSONAR cpp:S925
}
return R{};
return inOrderTraverseImpl<R, Node>(root, visitor, reverse);
}
template <typename Vis>
static auto traverse(Derived* const root, const Vis& visitor, const bool reverse = false)
static auto traverse(Derived* const root, const Vis& visitor, const bool reverse = false) //
-> std::enable_if_t<std::is_void<invoke_result<Vis, Derived&>>::value>
{
if (Node* const n = root)
{
Node::traverse<Vis>(down(n->lr[reverse]), visitor, reverse); // NOSONAR cpp:S925
visitor(*root);
Node::traverse<Vis>(down(n->lr[!reverse]), visitor, reverse); // NOSONAR cpp:S925
}
inOrderTraverseImpl<Node>(root, visitor, reverse);
}
template <typename Vis, typename R = invoke_result<Vis, const Derived&>>
static auto traverse(const Derived* const root, const Vis& visitor, const bool reverse = false) //
-> std::enable_if_t<!std::is_void<R>::value, R>
{
if (const Node* const n = root)
{
// NOLINTNEXTLINE(*-qualified-auto)
if (auto t = Node::traverse<Vis>(down(n->lr[reverse]), visitor, reverse)) // NOSONAR cpp:S925
{
return t;
}
if (auto t = visitor(*root)) // NOLINT(*-qualified-auto)
{
return t;
}
return Node::traverse<Vis>(down(n->lr[!reverse]), visitor, reverse);
}
return R{};
return inOrderTraverseImpl<R, const Node>(root, visitor, reverse);
}
template <typename Vis>
static auto traverse(const Derived* const root, const Vis& visitor, const bool reverse = false)
static auto traverse(const Derived* const root, const Vis& visitor, const bool reverse = false) //
-> std::enable_if_t<std::is_void<invoke_result<Vis, const Derived&>>::value>
{
if (const Node* const n = root)
{
Node::traverse<Vis>(down(n->lr[reverse]), visitor, reverse); // NOSONAR cpp:S925
visitor(*root);
Node::traverse<Vis>(down(n->lr[!reverse]), visitor, reverse); // NOSONAR cpp:S925
}
inOrderTraverseImpl<const Node>(root, visitor, reverse);
}

template <typename Vis>
static void postOrderTraverse(Derived* const root, const Vis& visitor, const bool reverse = false)
{
postOrderTraverseImpl<Node>(root, visitor, reverse);
}
template <typename Vis>
static void postOrderTraverse(const Derived* const root, const Vis& visitor, const bool reverse = false)
{
postOrderTraverseImpl<const Node>(root, visitor, reverse);
}
// NOLINTEND(misc-no-recursion)

private:
void rotate(const bool r) noexcept
Expand All @@ -256,6 +227,14 @@ class Node

auto retraceOnGrowth() noexcept -> Node*;

template <typename NodeT, typename DerivedT, typename Vis>
static void inOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse);
template <typename Result, typename NodeT, typename DerivedT, typename Vis>
static auto inOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse) -> Result;

template <typename NodeT, typename DerivedT, typename Vis>
static void postOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse);

void unlink() noexcept
{
up = nullptr;
Expand Down Expand Up @@ -527,6 +506,155 @@ auto Node<Derived>::retraceOnGrowth() noexcept -> Node*
return (nullptr == p) ? c : nullptr; // New root or nothing.
}

template <typename Derived>
template <typename NodeT, typename DerivedT, typename Vis>
void Node<Derived>::inOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse)
{
NodeT* node = root;
NodeT* prev = nullptr;

while (nullptr != node)
{
NodeT* next = node->up;

if (prev == node->up)
{
// We came down to this node from `prev`.

if (auto* left = node->lr[reverse])
{
next = left;
}
else
{
visitor(*down(node));

if (auto* right = node->lr[!reverse])
{
next = right;
}
}
}
else if (prev == node->lr[reverse])
{
// We came up to this node from the left child.

visitor(*down(node));

if (auto* right = node->lr[!reverse])
{
next = right;
}
}

prev = std::exchange(node, next);
}
}

template <typename Derived>
template <typename Result, typename NodeT, typename DerivedT, typename Vis>
auto Node<Derived>::inOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse) -> Result
{
NodeT* node = root;
NodeT* prev = nullptr;

while (nullptr != node)
{
NodeT* next = node->up;

if (prev == node->up)
{
// We came down to this node from `prev`.

if (auto* left = node->lr[reverse])
{
next = left;
}
else
{
if (auto t = visitor(*down(node))) // NOLINT(*-qualified-auto)
{
return t;
}

if (auto* right = node->lr[!reverse])
{
next = right;
}
}
}
else if (prev == node->lr[reverse])
{
// We came up to this node from the left child.

if (auto t = visitor(*down(node))) // NOLINT(*-qualified-auto)
{
return t;
}

if (auto* right = node->lr[!reverse])
{
next = right;
}
}

prev = std::exchange(node, next);
}
return Result{};
}

template <typename Derived>
template <typename NodeT, typename DerivedT, typename Vis>
void Node<Derived>::postOrderTraverseImpl(DerivedT* const root, const Vis& visitor, const bool reverse)
{
NodeT* node = root;
NodeT* prev = nullptr;

while (nullptr != node)
{
NodeT* next = node->up;

if (prev == node->up)
{
// We came down to this node from `prev`.

if (auto* left = node->lr[reverse])
{
next = left;
}
else if (auto* right = node->lr[!reverse])
{
next = right;
}
else
{
visitor(*down(node));
}
}
else if (prev == node->lr[reverse])
{
// We came up to this node from the left child.

if (auto* right = node->lr[!reverse])
{
next = right;
}
else
{
visitor(*down(node));
}
}
else
{
// We came up to this node from the right child.

visitor(*down(node));
}

prev = std::exchange(node, next);
}
}

/// This is a very simple convenience wrapper that is entirely optional to use.
/// It simply keeps a single root pointer of the tree. The methods are mere wrappers over the static methods
/// defined in the Node<> template class, such that the node pointer kept in the instance of this class is passed
Expand Down Expand Up @@ -612,6 +740,20 @@ class Tree final // NOSONAR cpp:S3624
return NodeType::template traverse<Vis>(*this, visitor, reverse);
}

/// Wraps NodeType<>::postOrderTraverse().
template <typename Vis>
void postOrderTraverse(const Vis& visitor, const bool reverse = false)
{
const TraversalIndicatorUpdater upd(*this);
NodeType::template postOrderTraverse<Vis>(*this, visitor, reverse);
}
template <typename Vis>
void postOrderTraverse(const Vis& visitor, const bool reverse = false) const
{
const TraversalIndicatorUpdater upd(*this);
NodeType::template postOrderTraverse<Vis>(*this, visitor, reverse);
}

/// Normally these are not needed except if advanced introspection is desired.
///
/// No linting and Sonar cpp:S1709 b/c implicit conversion by design.
Expand Down Expand Up @@ -640,7 +782,7 @@ class Tree final // NOSONAR cpp:S3624
return traverse([&i](const auto& x) { return (i-- == 0) ? &x : nullptr; }); // NOSONAR cpp:S881
}

/// Beware that this convenience method has linear complexity and uses recursion. Use responsibly.
/// Beware that this convenience method has linear complexity. Use responsibly.
auto size() const noexcept
{
auto i = 0UL;
Expand Down
Loading