qosd: сдавленные наброски по абстрактному дереву

This commit is contained in:
Сергей Маринкевич
2025-07-21 15:52:36 +07:00
commit 5df05f703a
20 changed files with 650 additions and 0 deletions
+43
View File
@@ -0,0 +1,43 @@
#pragma once
#include <vector>
#include <algorithm>
#include "ifaces/ILink.h"
#include "Logger.h"
/// \brief Базовая реализация интерфейса ILink для хранения связей между элементами.
/// \tparam TElem Тип элемента, между которыми устанавливается связь.
template <class TElem>
class BaseLink : public ILink<TElem> {
public:
using ElemPtr = std::shared_ptr<TElem>;
BaseLink(ElemPtr node) : owner_node_(node) {}
ElemPtr getNode() const override { return owner_node_.lock(); }
ElemPtr getParent() const override { return parent_.lock(); }
void setParent(const ElemPtr& parent) override { parent_ = parent; }
const std::vector<ElemPtr>& getChildren() const override { return children_; }
void addChild(const ElemPtr& child) override {
this->children_.push_back(child);
}
void removeChild(const ElemPtr& child) override {
children_.erase(std::remove(children_.begin(), children_.end(), child), children_.end());
}
void replaceChild(const ElemPtr& oldChild, const ElemPtr& newChild) override {
auto it = std::find(children_.begin(), children_.end(), oldChild);
if (it != children_.end()) {
*it = newChild;
}
}
~BaseLink() override {
Logger::get("Link").dbg("--- Destructor called for: BaseLink");
}
protected:
std::vector<ElemPtr> children_;
std::weak_ptr<TElem> owner_node_;
std::weak_ptr<TElem> parent_;
};
+12
View File
@@ -0,0 +1,12 @@
#pragma once
#include "links/BaseLink.h"
#include <stdexcept>
/// \brief Связь для листового узла, не допускающая дочерних элементов.
class LeafLink : public BaseLink {
public:
using BaseLink::BaseLink;
void addChild(const NodePtr&) override {
throw std::logic_error("LeafLink cannot have children");
}
};
+27
View File
@@ -0,0 +1,27 @@
#pragma once
#include <vector>
#include "ifaces/ILink.h"
#include "Logger.h"
/// \brief Заглушка для не реализованной связи между элементами.
/// Все методы возвращают пустые значения или не выполняют действий.
/// \tparam TElem Тип элемента.
template <class TElem>
class NotImplementedLink : public ILink<TElem> {
using ElemPtr = std::shared_ptr<TElem>>;
public:
NotImplementedLink(ElemPtr node) {}
ElemPtr getNode() const override { return nullptr; }
ElemPtr getParent() const override { return nullptr; }
void setParent(const ElemPtr& parent) override { }
const std::vector<ElemPtr>& getChildren() const override { return empty_; }
void addChild(const ElemPtr& child) override { }
void removeChild(const ElemPtr& child) override { }
~NotImplementedLink() override {
Logger::get("Link").dbg("--- Destructor called for: NotImplementedLink");
}
private:
std::vector<ElemPtr> empty_;
};
+17
View File
@@ -0,0 +1,17 @@
#pragma once
#include "links/BaseLink.h"
/// \brief Связь "один-ко-многим" между элементами одного типа.
/// Каждый дочерний элемент должен быть того же типа, что и родитель.
/// \tparam TElem Тип элемента.
template <class TElem>
class OneToManyLink : public BaseLink<TElem> {
public:
OneToManyLink(std::shared_ptr<TElem> e) : BaseLink<TElem>(e) {}
void addChild(const std::shared_ptr<TElem>& child) override {
/* Each child must be exactly the same type as parent */
if (typeid(*child) != typeid(*this->owner_node_.lock()))
throw std::logic_error("Foundling child");
BaseLink<TElem>::addChild(child);
}
};
+21
View File
@@ -0,0 +1,21 @@
#pragma once
#include "links/BaseLink.h"
#include <stdexcept>
/// \brief Связь "один-к-одному" между элементами.
/// Позволяет добавить только одного дочернего элемента.
/// \tparam TElem Тип элемента.
template <class TElem>
class OneToOneLink : public BaseLink<TElem> {
public:
using ElemPtr = std::shared_ptr<TElem>;
OneToOneLink(ElemPtr e) : BaseLink<TElem>(e) {}
void addChild(const ElemPtr& child) override {
if (!this->children_.empty())
throw std::logic_error("Too many children");
BaseLink<TElem>::addChild(child);
}
};