bump
This commit is contained in:
@@ -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>>;
|
||||||
+27
-25
@@ -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";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user