qosd: добавил краевые узлы
Краевые узлы не могут иметь дочерние узлы. Отлично подходит для бесклассовых дисциплин. Например, вариации FIFO, Fair Queue, RED и т.д. Псевдо-бесклассовые по типу TBF, которые подключают в себя FIFO (или любую указанную пользователем дисциплину), всё ещё должны пользоваться SimpleNode.
This commit is contained in:
@@ -3,10 +3,14 @@
|
|||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
/// \brief Связь для листового узла, не допускающая дочерних элементов.
|
/// \brief Связь для листового узла, не допускающая дочерних элементов.
|
||||||
class LeafLink : public BaseLink {
|
template <class TElem>
|
||||||
|
class LeafLink : public BaseLink<TElem> {
|
||||||
public:
|
public:
|
||||||
using BaseLink::BaseLink;
|
using ElemPtr = std::shared_ptr<TElem>;
|
||||||
void addChild(const NodePtr&) override {
|
|
||||||
throw std::logic_error("LeafLink cannot have children");
|
LeafLink(std::shared_ptr<TElem> e) : BaseLink<TElem>(e) {}
|
||||||
|
|
||||||
|
void addChild(const ElemPtr&) override {
|
||||||
|
throw std::logic_error("Leaf cannot have children");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,20 +1,21 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include "mixins/LazyLinkMixin.h"
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
|
#include "ifaces/INode.h"
|
||||||
|
#include "mixins/LazyLinkMixin.h"
|
||||||
#include "links/OneToManyLink.h"
|
#include "links/OneToManyLink.h"
|
||||||
#include "links/OneToOneLink.h"
|
#include "links/OneToOneLink.h"
|
||||||
#include "Logger.h"
|
#include "Logger.h"
|
||||||
|
|
||||||
/// \brief Миксин для иерархических связей между элементами.
|
/// \brief Миксин для иерархических связей между элементами.
|
||||||
/// \tparam TElem Тип дочернего элемента.
|
/// \tparam INode Тип дочернего элемента.
|
||||||
/// Автоматически выбирает тип связи (один-ко-многим или один-к-одному) в зависимости от типа
|
/// Автоматически выбирает тип связи (один-ко-многим или один-к-одному) в зависимости от типа
|
||||||
/// дочернего узла. Если тип дочернего узла совпадает с родителем, то используется связь
|
/// дочернего узла. Если тип дочернего узла совпадает с родителем, то используется связь
|
||||||
/// один-ко-многим. При попытке подключить узел отличного типа выбирается связь один-к-одному.
|
/// один-ко-многим. При попытке подключить узел отличного типа выбирается связь один-к-одному.
|
||||||
template <class TElem>
|
class HierarchicalLinkMixin : public LazyLinkMixin<OneToOneLink<INode>> {
|
||||||
class HierarchicalLinkMixin : public LazyLinkMixin<OneToOneLink<TElem>> {
|
using LinkPtr = std::shared_ptr<ILink<INode>>;
|
||||||
using LinkPtr = std::shared_ptr<ILink<TElem>>;
|
using ElemPtr = std::shared_ptr<INode>;
|
||||||
using ElemPtr = std::shared_ptr<TElem>;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
~HierarchicalLinkMixin() override {
|
~HierarchicalLinkMixin() override {
|
||||||
@@ -23,7 +24,7 @@ public:
|
|||||||
|
|
||||||
void linkChild(const ElemPtr& child) override {
|
void linkChild(const ElemPtr& child) override {
|
||||||
hierarchicalInit(child);
|
hierarchicalInit(child);
|
||||||
LazyLinkMixin<OneToOneLink<TElem>>::linkChild(child);
|
LazyLinkMixin<OneToOneLink<INode>>::linkChild(child);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -36,10 +37,10 @@ protected:
|
|||||||
|
|
||||||
if (typeid(*child) == typeid(*this)) {
|
if (typeid(*child) == typeid(*this)) {
|
||||||
Logger::get("Mixin").dbg("--- Mutate to OneToMany");
|
Logger::get("Mixin").dbg("--- Mutate to OneToMany");
|
||||||
newLink = std::make_shared<OneToManyLink<TElem>>(*this);
|
newLink = std::make_shared<OneToManyLink<INode>>(*this);
|
||||||
} else {
|
} else {
|
||||||
Logger::get("Mixin").dbg("--- Mutate to OneToOne");
|
Logger::get("Mixin").dbg("--- Mutate to OneToOne");
|
||||||
newLink = std::make_shared<OneToOneLink<TElem>>(*this);
|
newLink = std::make_shared<OneToOneLink<INode>>(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newLink && this->link_)
|
if (newLink && this->link_)
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
/// \brief Класс сложного (составного) узла дерева.
|
/// \brief Класс сложного (составного) узла дерева.
|
||||||
/// Может содержать несколько дочерних ComplexNode или один SimpleNode.
|
/// Может содержать несколько дочерних ComplexNode или один SimpleNode.
|
||||||
class ComplexNode : public BaseNode,
|
class ComplexNode : public BaseNode,
|
||||||
virtual public HierarchicalLinkMixin<INode>,
|
virtual public HierarchicalLinkMixin,
|
||||||
public FabricMixin<ComplexNode> {
|
public FabricMixin<ComplexNode> {
|
||||||
public:
|
public:
|
||||||
~ComplexNode() {
|
~ComplexNode() {
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "nodes/BaseNode.h"
|
||||||
|
#include "mixins/LazyLinkMixin.h"
|
||||||
|
#include "mixins/FabricMixin.h"
|
||||||
|
#include "links/LeafLink.h"
|
||||||
|
#include "Logger.h"
|
||||||
|
|
||||||
|
/// \brief Класс простого (листового) узла дерева.
|
||||||
|
/// Может содержать только одного дочернего ComplexNode.
|
||||||
|
class LeafNode : public BaseNode,
|
||||||
|
virtual public LazyLinkMixin<LeafLink<INode>>,
|
||||||
|
public FabricMixin<LeafNode> {
|
||||||
|
public:
|
||||||
|
~LeafNode() {
|
||||||
|
Logger::get("ConDes").dbg(std::string("--- Leaf destructor called for: ") + name_);
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
friend class FabricMixin<LeafNode>;
|
||||||
|
LeafNode(std::string name) : BaseNode(std::move(name)) {
|
||||||
|
Logger::get("ConDes").dbg(std::string("--- Leaf constructor called for: ") + name_);
|
||||||
|
}
|
||||||
|
};
|
||||||
+11
-1
@@ -4,6 +4,7 @@
|
|||||||
|
|
||||||
#include "nodes/SimpleNode.h"
|
#include "nodes/SimpleNode.h"
|
||||||
#include "nodes/ComplexNode.h"
|
#include "nodes/ComplexNode.h"
|
||||||
|
#include "nodes/LeafNode.h"
|
||||||
#include "iterators/Traversal.h"
|
#include "iterators/Traversal.h"
|
||||||
|
|
||||||
|
|
||||||
@@ -58,9 +59,18 @@ int main() {
|
|||||||
auto child3 = ComplexNode::create("ComplexChild3");
|
auto child3 = ComplexNode::create("ComplexChild3");
|
||||||
root->linkChild(child3);
|
root->linkChild(child3);
|
||||||
|
|
||||||
child3->linkChild(SimpleNode::create("SimpleSubChild3"));
|
auto leaf3 = LeafNode::create("LeafSubChild3");
|
||||||
|
child3->linkChild(leaf3);
|
||||||
|
|
||||||
{
|
{
|
||||||
|
// Негативный сценарий 0: попытка добавить SimpleNode к LeafNode
|
||||||
|
try {
|
||||||
|
leaf3->linkChild(SimpleNode::create("ShouldFail"));
|
||||||
|
logger.err("[ERROR] Не должно было получиться добавить SimpleNode к LeafNode!");
|
||||||
|
} catch (const std::logic_error& e) {
|
||||||
|
logger.warn(std::string("[Ожидаемое исключение] ") + e.what());
|
||||||
|
}
|
||||||
|
|
||||||
// Негативный сценарий 1: попытка добавить второй SimpleNode к ComplexNode
|
// Негативный сценарий 1: попытка добавить второй SimpleNode к ComplexNode
|
||||||
try {
|
try {
|
||||||
child2->linkChild(SimpleNode::create("ShouldFail"));
|
child2->linkChild(SimpleNode::create("ShouldFail"));
|
||||||
|
|||||||
Reference in New Issue
Block a user