iters
Sergey Marinkevich 6 months ago
parent 0f93988fb6
commit 038cbb73f4

@ -5,7 +5,6 @@
template <typename T> template <typename T>
class IIterator { class IIterator {
public: public:
virtual T& operator*() const = 0;
virtual T* operator->() const = 0; virtual T* operator->() const = 0;
virtual operator const T*() const = 0; virtual operator const T*() const = 0;

@ -0,0 +1,42 @@
#pragma once
#include <stack>
#include "iterators/BaseIterator.h"
template <typename T>
class BFSIterator : public BaseIterator<T> {
public:
explicit BFSIterator(T* ptr) : BaseIterator<T>(ptr) {
if (!this->current)
return;
for (auto& it : BaseIterator<T>::current->children())
q.push(it.get());
}
IIterator<T>& operator++() override {
advance();
return *this;
}
virtual size_t level() const {
return level_;
}
protected:
std::queue<T*> q;
size_t level_;
void advance() {
if (q.empty()) {
this->current = nullptr;
return;
}
this->current = q.front();
q.pop();
for (auto& it : this->current->children())
q.push(it.get());
}
};

@ -1,23 +1,43 @@
#pragma once #pragma once
#include <tuple>
#include "ifaces/IIterator.h" #include "ifaces/IIterator.h"
/// \brief Базовый итератор. /// \brief Базовый итератор.
/// \tparam T Тип итерируемого элемента. /// \tparam T Тип итерируемого элемента.
/// Предоставляет базовую реализацию методов для работы с итерируемыми объектами. /// Предоставляет базовую реализацию методов для работы с итерируемыми объектами.
/// Возвращает объект-прокси, мимикрирующий под итерируемый элемент. Позволяет использовать
/// распаковку структуры для получения уровня элемента в поддереве.
template <typename T> template <typename T>
class BaseIterator : public IIterator<T> { class BaseIterator : public IIterator<T> {
protected:
T* current;
public: public:
explicit BaseIterator(T* ptr) : current(ptr) {} explicit BaseIterator(T* ptr) : current(ptr) {}
T& operator*() const override { return *current; } struct Proxy {
T* e;
size_t level;
T& operator*() const { return *e; }
T* operator->() const { return e; }
};
Proxy& operator*() {
proxy = Proxy{ current, this->level() };
return proxy;
}
T* operator->() const override { return current; } T* operator->() const override { return current; }
virtual operator const T*() const { return current; } operator const T*() const { return current; }
operator T&() const { return *current; }
operator T*() const { return current; }
bool operator!=(const IIterator<T>& other) const override { bool operator!=(const IIterator<T>& other) const override {
const T* other_current = other; const T* other_current = other;
return current != other_current; return current != other_current;
} }
protected:
T* current;
private:
Proxy proxy;
}; };

@ -1,17 +1,18 @@
#pragma once #pragma once
#include <stack> #include <stack>
#include <iterator>
#include "iterators/BaseIterator.h" #include "iterators/BaseIterator.h"
template <typename T> template <typename T>
class DFSIterator : public BaseIterator<T> { class DFSIterator : public BaseIterator<T> {
public: public:
explicit DFSIterator(T* ptr) : BaseIterator<T>(ptr) { explicit DFSIterator(T* ptr) : BaseIterator<T>(ptr), level_(0) {
if (!this->current) if (!this->current)
return; return;
for (auto& it : BaseIterator<T>::current->children()) for (auto it = BaseIterator<T>::current->children().rbegin(); it != BaseIterator<T>::current->children().rend(); ++it)
s.push(it.get()); s.push({it->get(), 1});
} }
IIterator<T>& operator++() override { IIterator<T>& operator++() override {
@ -22,8 +23,9 @@ public:
virtual size_t level() const { virtual size_t level() const {
return level_; return level_;
} }
protected: protected:
std::stack<T*> s; std::stack<std::pair<T*, size_t>> s;
size_t level_; size_t level_;
void advance() { void advance() {
@ -32,10 +34,14 @@ protected:
return; return;
} }
this->current = s.top(); auto [node, lvl] = s.top();
s.pop(); s.pop();
for (auto& it : this->current->children()) this->current = node;
s.push(it.get()); level_ = lvl;
for (auto it = this->current->children().rbegin();
it != this->current->children().rend(); ++it)
s.push({it->get(), lvl + 1});
} }
}; };

@ -0,0 +1,22 @@
#pragma once
#include <queue>
#include "ifaces/INode.h"
#include "iterators/DFSIterator.h"
#include "iterators/BFSIterator.h"
template<typename Policy>
class Traversal {
INode* start;
public:
explicit Traversal(INode* start) : start(start) {}
explicit Traversal(NodePtr start) : start(start.get()) {}
Policy begin() const { return Policy(start); }
Policy end() const { return Policy(nullptr); }
};
using TraversalDFS = Traversal<DFSIterator<INode>>;
using TraversalBFS = Traversal<BFSIterator<INode>>;

@ -4,32 +4,26 @@
#include "nodes/SimpleNode.h" #include "nodes/SimpleNode.h"
#include "nodes/ComplexNode.h" #include "nodes/ComplexNode.h"
#include "iterators/DFSIterator.h" #include "iterators/Traversal.h"
template<typename Policy> void printNode(INode& node, size_t level) {
class Traversal { for (size_t i = level; i > 0; i--)
INode* start;
public:
explicit Traversal(INode* start) : start(start) {}
Policy begin() const { return Policy(start); }
Policy end() const { return Policy(nullptr); }
};
void printNode(INode& node) {
NodePtr parent = node.parent();
while (parent) {
std::cout << " "; std::cout << " ";
parent = parent->parent();
}
std::cout << node.name() << "\n"; std::cout << node.name() << "\n";
} }
void printTree(INode& node) { void printTreeBFS(INode& node) {
for (auto& child : Traversal<DFSIterator<INode>>(&node)) for (auto& child : TraversalBFS(&node))
printNode(child); printNode(*child, 0);
}
void printTreeList(INode& node) {
for (auto& child : TraversalDFS(&node))
printNode(*child, 0);
}
void printTreeLadder(INode& node) {
for (auto& [child, level] : TraversalDFS(&node))
printNode(*child, level);
} }
int main() { int main() {
@ -47,7 +41,7 @@ int main() {
root->linkChild(child1); root->linkChild(child1);
std::cout << "\nInit tree:\n"; std::cout << "\nInit tree:\n";
printTree(*root); printTreeLadder(*root);
std::cout << "\n"; std::cout << "\n";
{ {
@ -99,21 +93,29 @@ int main() {
} }
std::cout << "\nAnother tree:\n"; std::cout << "\nAnother tree:\n";
printTree(*root); printTreeLadder(*root);
std::cout << "\n";
std::cout << "\nList:\n";
printTreeList(*root);
std::cout << "\n";
std::cout << "\nBFS:\n";
printTreeBFS(*root);
std::cout << "\n"; std::cout << "\n";
logger.info("Unlinking ComplexChild2...\n"); logger.info("Unlinking ComplexChild2...\n");
child2->unlinkParent(); child2->unlinkParent();
std::cout << "\nTree after unlink:\n"; std::cout << "\nTree after unlink:\n";
printTree(*root); printTreeLadder(*root);
std::cout << "\n"; std::cout << "\n";
logger.info("Put refs of ComplexChild2, ComplexChild3 and its children"); logger.info("Put refs of ComplexChild2, ComplexChild3 and its children");
} }
std::cout << "\nTree after scope out:\n"; std::cout << "\nTree after scope out:\n";
printTree(*root); printTreeLadder(*root);
std::cout << "\n"; std::cout << "\n";
logger.info("Unlinking ComplexChild1 and ComplexChild3...\n"); logger.info("Unlinking ComplexChild1 and ComplexChild3...\n");
@ -130,7 +132,7 @@ int main() {
root->linkChild(child4); root->linkChild(child4);
std::cout << "\nTree flush and link SimpleChild4:\n"; std::cout << "\nTree flush and link SimpleChild4:\n";
printTree(*root); printTreeLadder(*root);
std::cout << "\n"; std::cout << "\n";
} }

Loading…
Cancel
Save