Введение и постановка задачи
Мы пройдем по следующим шагам:
Посмотрим на проблему.
Решим ее обычным полиморфизмом.
-
Пройдем небольшими шагами к более элегантному решению
Strategy
External Polymorphism
Type Erasure
Немного философии.
Изменения - основная проблема создания программного обеспечения. Большая задача в разработке ПО - сделать его таким, чтобы его можно было легко изменять. И одной из основных проблем в изменении существующего кода - его зависимости.
Привязанность кода к внешним зависимостям очень сильно "сковывает" движения программиста при внесении чего-то нового в код.
Посмотрим на задачу: хотим рисовать разные фигуры.
Задача искусственная, но очень хорошо иллюстрирующая всё, что я собираюсь показать.
Но не заостряйте свое внимание именно на фигурах, все-таки паттерн, про который я собираюсь рассказать, применим еще много к чему.
Наследование
Мы хотим рисовать разные фигуры. Сперва создадим базовый абстрактный класс Shape
с методом его рисования на экран
struct Shape {
virtual void draw() = 0;
};
Теперь определим парочку конкретных фигур, например квадрат и круг унаследовав их классы Square
и Circle
от Shape
.
Также в них определим виртуальный метод draw
.
struct Circle : public Shape {
void draw() override {
std::cout << "I am Circle" << std::endl;
}
};
struct Square : public Shape {
void draw() override {
std::cout << "I am Square" << std::endl;
}
};
Типов фигур может быть очень много, каждый из классов фигур обязан давать определение методу draw
. Но такой подход может очень быстро привести нас к проблемам.
Рассмотрим конкретный класс Circle
- в нашей текущей реализации он сильно связан с деталями реализации механизма рисования фигур на экран, а это плохо.
Например, реализация рисования фигур может быть зашита где-то глубоко в используемой библиотеке или написана в другом месте проекта.
К тому же такой подход позволяет иметь только одну реализацию рисования фигур. Допустим, мы пишем приложение, используя OpenGL для рисования разной информации на экран, но вдруг нам понадобилось портировать весь рисующий функционал ещё и на Vulkan/Metal/DirectX. Что делать в таком случае? К решению этой проблемы можно подойти с разных сторон.
Первый подход - добавить новые методы рисования:
struct Shape {
virtual void drawOpenGL() = 0;
virtual void drawVulkan() = 0;
// и так далее
};
Тогда при использовании данного класса нам нужно будет делать выбор, какой из методов использовать:
void drawAll(std::vector<Shape*> v){
for(auto *shape: v){
switch (API::getGraphicsApi()) {
case OpenGL:
shape->drawOpenGL();
break;
case Vulkan:
shape->drawVulkan();
break;
default:
throw std::runtime_error("unsupported graphics api");
}
}
}
Второй подход - создать новые классы для каждого из графических движков.
struct CircleOpenGL : public Circle {
void draw() override {
std::cout << "I am Circle (OpenGL)" << std::endl;
}
};
struct CircleVulkan : public Circle {
void draw() override {
std::cout << "I am Circle (Vulkan)" << std::endl;
}
};
Тогда при создании новых объектов фигур нужно будет откуда-то узнавать, поддержка какого графического движка есть на текущей машине и создавать объект соотвествующего класса:
Shape* createCircle(){
switch (API::getGraphicsApi()) {
case OpenGL:
return new CircleOpenGL;
break;
case Vulkan:
return new CircleVulkan;
break;
default:
throw std::runtime_error("unsupported graphics api");
}
}
И аналогично для Square
. Да, оба подхода сработают. Для маленьких проектов, возможно, даже ничего страшного не произойдёт.
Но проекты развиваются. Представим, что через какое-то время нам потребовалось сохранять существующие в программе фигуры в постоянную память. Требуется создать новый метод serialize
. Добавляем его в базовый класс:
struct Shape {
virtual void draw() = 0;
virtual void serialize() = 0;
};
И вдруг понимаем, что делать сериализацю объектов можно в множество разных форматов (JSON, toml, XML, ...). И опять та же история, что и с разными графическими движками. Я уже не буду описывать, что иметь в своей программе подобные классы - плохо:
struct CircleOpenGL_JSON : public Circle {/* */};
struct CircleOpenGL_XML : public Circle {/* */};
struct CircleVulkan_JSON : public Circle {/* */};
struct CircleVulkan_XML : public Circle {/* */};
Стоит отметить, что добавление нового метода в базовый класс привело нас также к дублированию кода. Метод draw
будет одинаковым у классов CircleOpenGL_JSON
и CircleOpenGL_XML
, а метод serialize
будет одинаковым у классов CircleOpenGL_JSON
и CircleVulkan_JSON
.
Иерахия классов становится всё глубже и запутаннее. А если нам понадобится ещё один метод в базовом классе?
Диаграмма этого ужаса:
Результаты такого подхода:
Очень много наследования
Нелепые имена классов
Огромные иерархии классов
Дублирования кода (DRY)
Невероятно сложное добавления нового функционала
Сложность сопровождения кода
В попытках справиться с недостатками такого подхода разработчик может сделать всё ещё хуже.
Паттерны
Предыдущий подход был наивным, не имеющим возможности жить в больших проектах.
Рассмотрим более современный подход - использование паттернов.
Паттерн:
Имеет имя
Одним лишь своим именем объясняет уже многое
Нацелен на уменьшение связности
Предоставляет своего рода абстракцию
Проверен временем
В тот момент, когда мы добавляли в классы фигур уже второй метод (serialize
), этот паттерн уже витал где-то поблизости. И это паттерн стратегия.
Суть паттерна: Источник
Стратегия — это поведенческий паттерн проектирования, который определяет семейство схожих алгоритмов и помещает каждый из них в собственный класс, после чего алгоритмы можно взаимозаменять прямо во время исполнения программы.
Создадим отдельный класс, определяющий поведение при рисовании фигуры:
class DrawStrategy{
virtual void draw(Circle*) = 0;
virtual void draw(Square*) = 0;
};
Теперь мы можем определить несколько разных методов рисования фигур, унаследовавшись от DrawStrategy
:
class DrawStrategyOpenGL : public DrawStrategy{
void draw(Circle* circle) override {
// do OpenGL stuff
}
void draw(Square* square) override {
// do OpenGL stuff
}
};
class DrawStrategyVulkan : public DrawStrategy{
/* аналогично */
};
А в классе Circle
теперь добавим поле, хранящее метод его рисования.
struct Circle : public Shape {
std::unique_ptr<DrawStrategy> drawStrategy;
explicit Circle(DrawStrategy *drawStrategy)
: drawStrategy(drawStrategy) {
}
void draw() override {
drawStrategy->draw(this);
}
};
Применение:
int main(){
std::vector<std::unique_ptr<Shape>> v;
v.emplace_back(
std::make_unique<Circle>(std::make_unique<DrawStrategyOpenGL>())
);
v.emplace_back(
std::make_unique<Circle>(std::make_unique<DrawStrategyVulkan>())
);
for(auto &sh: v){
sh->draw();
}
}
Таким образом мы выделили в отдельный класс метод рисования. В общем случае - тот метод, который может в разных обстоятельствах меняться.
Теперь, если потребуется добавить поддержку нового графического движка. то нужно будет всего лишь добавить новый класс, унаследованный от DrawStrategy
. Не придется менять уже существующий код.
Диаграмма:
В целом применение паттерна стратегия позволило нам разделить нашу программу на 3 разных уровня. Вот они, отсортированные по их абстрактности:
Класс
Shape
- высокоуровневый интерфейсКлассы
Circle
иSquare
- реализации интерфейса (средний уровень)Абстрактный класс
DrawStrategy
и его наследники - вынесенное в отдельное место поведение (низкоуровневая реализация)
Данный паттерн не ограничивается лишь тем, что наш класс содержит какой-то внешний объект, содержащий в себе реализацю.метода. Стратегии также распространены и вне мира ООП, например:
std::vector<int> numbers = {1, 2, 3, 4, 5};
// хотим посчитать сумму чисел
std::accumulate(numbers.cbegin(), numbers.cend(),
0,
std::plus{} // <-- СТРАТЕГИЯ: "складывать числа"
);
// хотим посчитать произведение чисел
std::accumulate(numbers.cbegin(), numbers.cend(),
1,
std::multiplies{} // <-- СТРАТЕГИЯ: "умножать числа"
);
Стратегии в стандартной библиотеке шаблонов:
template<
typename _Tp,
typename _Alloc = std::allocator<_Tp>> // <-- СТРАТЕГИЯ.
//Аллокатор определяет, как будет выделяться память
class vector {/* ... */};
template<
typename _Value,
typename _Hash = hash<_Value>, // <-- СТРАТЕГИЯ Хеш функция
typename _Pred = equal_to<_Value>, // <-- СТРАТЕГИЯ Как сравнивать ключи
typename _Alloc = allocator<_Value>> // <-- СТРАТЕГИЯ (см. выше)
class unordered_set {/* ... */};
template <
typename _Tp,
typename _Dp = default_delete<_Tp>> // <-- СТРАТЕГИЯ Как освобождать память
class unique_ptr
Итоги применения паттерна стратегия:
Вынесение деталей реализации в отдельные классы. (Принцип единственной ответственности/Single responsibility principle)
Создали возможность легкого расширения (Принцип открытости/закрытости. OCP)
Разделили интерфейсы (Interface segregation principle)
Избавились от дублирования кода (DRY)
Избавились от глубины иерархии
Упростили сопровождение кода. Легче понимать, легче писать
Но! Минусы:
Производительность с точки зрения вызовов. При вызове
draw
происходит на самом деле два вызова:main
->Circle::draw
->DrawStrategy::draw(Circle*)
Производительность с точки зрения памяти. Множество маленьких выделений памяти для стратегий.
Производительность с точки зрения указателей. Много указателей, мы переходим по их адресам.
Нужно создавать отдельные абстрактные классы-стратегии для другой функциональности, например
SerializeStrategy
для сериализации.Если отказаться от умных указателей в пользу производительности, то придётся вручную управлять временем жизни объектов. См. Интересная лекция про цену абстракций.
Circle
иSquare
всё еще знают про то, что их нужно как-то рисовать. Они всё еще несут некоторую ответственность за эти операции. Что-то в этом чувствуется не так. Операция рисования, конечно, зависит от фигуры, которая рисуется в данный момент. Но по идее самой фигуре не должно быть дела, рисуют ли её или делают что-то другое. Это слегка размывает абстракцию фигуры.
Существует решение лучше!
Двигаемся к Type Erasure
Вы уже могли слышать что-то про стирание типа, поэтому уточню:
-
Это НЕ про
void*
указатели на базовый класс
std::variant
.std::variant
основан на фиксированном наборе типов и предоставляет открытый набор операций над ними. Мы же пытаемся достигнуть обратного - открытого для расширения набора типов и фиксированного набора операций над ними.
-
Это про
Шаблонный конструктор
Интерфейс без единого слова
virtual
Смесь паттернов External Poymorphism, Bridge, Prototype
Посмотрим на класс Circle
, ещё не испорченный разными не относящимися к фигурам методами, а также наследованием:
class Circle {
public:
explicit Circle(double r)
: radius(r) {}
double getRadius() const {
return radius;
}
void setRadius(double r) {
radius = r;
}
private:
double radius;
// тут могут быть еще полезные данные
// координаты центра например
};
И аналогично может быть определён класс Square
.
Этим классам не нужен базовый класс
Им не нужно знать друг о друге
Они не должны заботиться о том, что с ними можно сделать
Это очень удобно. Такими классами максимально просто пользоваться. У них нет никаких зависимостей. И главное - мы их больше никогда не изменим!
Теперь перейдем к решению проблемы их рисования.
Паттерн External Polymorphism
Описание задачи этого паттерна из исходного документа, описывающего его:
Allow classes that are not related by inheritance and/or have no virtual methods to be treated polymorphically.
И мой вольный перевод:
Дать возможность классам, не связанным наследованием и/или не имеющим виртуальных методов, быть обработанными так, как будто они полиморфные.
Посмотрим на код и разберем, что к чему.
struct ShapeConcept {
virtual ~ShapeConcept() = default;
};
template<typename T>
struct ShapeModel : public ShapeConcept {
T object;
explicit ShapeModel(T &&shape) : object(std::move(shape)) {}
explicit ShapeModel(const T &shape) : object(shape) {}
};
Чуть позже станет ясно, почему здесь написано struct
, а не class
.
Конструктор ShapeModel
принимает объект любого класса и сохраняет его в своё поле. Этим классом может быть Circle
или Square
, или любая другая фигура, которую мы создадим.
В то же время ShapeModel
наследуется от ShapeConcept
, чуть позже станет ясно, почему.
Теперь в ShapeConcept
добавим все функции-операции над фигурами, которые могут быть нужны нам (оставлю в будущем только draw
для краткости).
struct ShapeConcept {
virtual ~ShapeConcept() = default;
virtual void draw() const = 0;
// ...
};
И особенным образом дадим определение этим функциям в производном классе ShapeModel
:
template<typename T>
struct ShapeModel : public ShapeConcept {
T object;
explicit ShapeModel(T &&shape) : object(std::move(shape)) {}
explicit ShapeModel(const T &shape) : object(shape) {}
void draw() const override {
draw(object); // Что за функция draw? См. после кода.
}
};
Что же за вызов функции draw(object)
? Когда мы пишем такое, мы утверждаем, что где-то вне классов, только что созданных нами, существует функция draw
, которая в качестве аргумента сможет принять object
типа T
. Например: void draw(const Circle &c) { ... }
Это уже наложило ограничения на то, какие типы могут быть подставлены вместо T
во время инстанциации шаблона ShapeModel
. А именно: для этих типов обязательно должна существовать функция draw
, принимающая их в качестве аргумента.
ShapeModel
наследуется от ShapeConcept
. Таким образом, с помощью задания чисто виртуальных функций в классе ShapeConcept
мы говорим, какие функции-обработчики должны обязательно существовать для объектов, которые мы будем в будущем хранить внутри ShapeModel
.
В этом и заключается паттерн External Polymorphism.
Мы извлекли полиморфную часть классов иерархии(которая теперь уже и не нужна) в отдельное место
Мы всё еще можем строго задавать функции-обработчики наших классов
Всё еще могут существовать абстрактные классы (для которых нет полного набора функций-обработчиков)
В итоге данный паттерн:
Позволяет обрабатывать любой объект так, как будто он полиморфный. Можно даже создать видимость полиморфизма для фундаментальных типов (например
int
)Позволяет вынести детали реализации из класса
Позволяет классам не заботиться (и даже знать) о тех операциях, которые над ними совершаются
Открывает возможность лёгкого расширения функциональности
Покажу пример работы:
class Circle { /* см. выше */ };
struct ShapeConcept { /* см. выше */ };
template<typename T>
struct ShapeModel : public ShapeConcept { /* см. выше */ };
// Определим функцию, которая умеет рисовать круг
void draw(const Circle &s) {
std::cout << "I am Circle with radius = " <<
s.getRadius() << std::endl;
}
// Теперь класс Circle отвечает требованиям ShapeConcept
int main(){
std::vector<std::unique_ptr<ShapeConcept>> v;
v.emplace_back(std::make_unique<ShapeModel<Circle>>{3.0});
v.emplace_back(std::make_unique<ShapeModel<Circle>>{4.0});
v.emplace_back(std::make_unique<ShapeModel<Circle>>{5.0});
for(auto *shape: v){
shape->draw();
}
// int не отвечает требованиям ShapeConcept
// следующий код не скомпилируется
// ошибка при инстанциации класса ShapeModel<int>
// так как нет функци draw(int)
ShapeModel test(123);
}
Из оригинального документа, описывающего данный паттерн есть ещё один пример. Допустим, мы работаем с несколькими библиотеками и хотим написать сериализацию для объектов из этих библиотек. Взять и создать для них общий базовый класс не представляется возможным. С помощью данного паттерна мы можем решить эту проблему.
Создаем класс SerializableConcept
, который описывает, какие внешние функции должны существовать (serialize
в данном случае). От этого класса наследуем класс SerializableModel
(аналогично ShapeModel
). И получаем возможность создать SerializableModel
над любым классом, для которого существует функция serialize
, принимающая этот класс как аргумент. И, соответственно, теперь мы можем сериализовать любой класс.
И... Ура! Это работает! Но.
Подведём итоги применения паттерна External Polymorphism:
Много указателей
Много
std::make_unique
И много других вещей, которые мы не хотим делать вручную
Давайте улучшим этот паттерн, чтобы избавиться от данных проблем.
Type Erasure
Возьмём и обернем ShapeConcept
и ShapeModel
в класс Shape
, в его секцию private
(становится понятно, зачем было объявлять их struct
- теперь они всё равно спрятаны).
class Shape {
private:
struct ShapeConcept { /* см. выше */ };
template<typename T>
struct ShapeModel : public ShapeConcept { /* см. выше */ };
};
Таким образом мы прячем эти относительно искусственные два класса с не очень говорящими именами внутрь уже довольно очевидной оболочки.
Теперь предоставим пользователю возможность создать любую фигуру. Добавим в класс Shape
поле, хранящее конкретную фигуру и шаблонный конструктор:
class Shape {
private:
struct ShapeConcept { /* см. выше */ };
template<typename T>
struct ShapeModel : public ShapeConcept { /* см. выше */ };
std::unique_ptr<ShapeConcept> shapePtr; // новое поле
public:
template<typename T>
explicit Shape(T &&shape)
: shapePtr(new ShapeModel<T>(std::forward(shape))) {}
};
А теперь присмотритесь. Что делает этот новый конструктор? Он создает для переданного объекта соответсвующую ему ShapeModel
Заметьте, новое поле имеет тип std::unique_ptr<ShapeConcept>
и конструктор сохраняет в него указатель на ShapeModel
. А теперь посмотрим на то, что происходит с типами во время всех этих действий:
Объект передан в конструктор
Shape
- тип известен -T
Создан объект класса
ShapeModel<T>
- тип исходного объекта всё еще здесьУказатель на
ShapeModel<T>
сохранен внутри указателя наShapeConcept
. Так можно сделать, ведь они связаны наследованием. Но! Указатель наShapeConcept
уже не содержит в себе типа, который лежит внутри него. На этом шаге произошло "стирание типа". Отсюда и название данного паттерна.
Но несмотря на то, что тип кажется утерянным, мы всё еще можем пользоваться объектом, который только что сохранили. Всё, что нужно для его обработки уже написано в классах ShapeConcept
и ShapeModel
. Нужна лишь внешняя функция-обработчик.
Этот шаг с созданием шаблонного конструктора Shape
и есть проявляение паттерна Bridge.
Для справки (wikipedia): Мост - структурный шаблон проектирования, используемый в проектировании программного обеспечения чтобы «разделять абстракцию и реализацию так, чтобы они могли изменяться независимо»
Теперь мы можем создать сколько нам угодно классов для разных фигур, создать перегрузки функции draw
, способные принимать объекты их типов в качестве аргументов. А красота в том, что компилятор инстанциирует нужные шаблоны и предоставит нам возможность пользоваться ими, как полиморфными! Нам самим не нужно писать этот код.
И последнее, что нужно для пользования классом Shape
. Все еще нет возможности нарисовать сохраненную фигуру. Для решения этой проблемы создадим функцию:
class Shape {
private:
/* см. выше */
public:
/* конструктор */
friend void draw(const Shape &shape) {
shape.shapePtr->draw();
// shape - Тип Shape
// shape.shapePtr - Тип ShapeConcept
// shape.shapePtr->draw() -> Вызов ShapeModel::draw()
// Внутри реализации
// Вызов ShapeModel::draw() -> Вызов draw(T)
}
};
Новая функция должна быть объявлена со словом friend
, чтобы она могла внутри себя обратиться к полю shapePtr
класса Shape
.
Пример применения:
int main(){
Shape circle(Circle{3.14});
draw(circle); // красиво!
}
Теперь этим можно пользоваться, рисовать разные фигуры. Но, скорее всего, в мы столкнемся с проблемой. Как копировать объекты класса Shape
? Ведь этот класс не знает тип объекта, который он хранит внутри себя.
С решением этой проблемы поможет паттерн Прототип.
Суть паттерна (wikipedia): Прототип — это порождающий паттерн проектирования, который позволяет копировать объекты, не вдаваясь в подробности их реализации.
Просто добавим новый чисто виртуальный метод clone
в ShapeConcept
:
struct ShapeConcept {
virtual ~ShapeConcept() = default;
virtual std::unique_ptr<ShapeConcept> clone() const = 0;
virtual void draw() const = 0;
// ...
};
Теперь этот новый метод отвечает за копирование (клонирование) объектов.
Дадим этому методу определение в ShapeModel
:
template<typename T>
struct ShapeModel : public ShapeConcept {
/* всё остальное */
std::unique_ptr<ShapeConcept> clone() const override {
return std::make_unique<ShapeModel>(*this);
}
};
Конечно же, дадим пользователю возможность использовать этот метод, добавим соотвествующую функцию в Shape
:
class Shape {
private:
/* см. выше */
std::unique_ptr<ShapeConcept> shapePtr;
public:
/* всё остальное */
Shape(const Shape& other)
: shapePtr(other.shapePtr->clone()) {}
};
Теперь можно написать так:
int main(){
Shape circle(Circle{3.14});
auto circle_copy = circle;
draw(circle_copy);
}
Анализ данного решения:
Shape
- высокоуровневая абстракция "контейнера" для фигур, который позволяет хранить в себе только те фигуры, которые соответствуют требованиям ShapeConcept.Circle
,Square
и др. - содержат только нужную информацию. Не знают о классеShape
. Не знают об операциях над ними. Не связаны полиморфизмом. Среднеуровневая абстракция.Функции
draw
,serialize
и др. Делают некие операции над нужными фигурами - низкоуровневая абстракция.Класс
ShapeModel
- хранит в себе конкретную фигуру, связываетShape
с конкретными функциями-обработчиками фигур (Мост). Сгенерирован компилятором.
Что мы получили:
Извлекли детали реализации
Предоставили возможность лёгкого расширения функциональности
Разделение интерфейсов
Отсутствие дублирования кода
Классы, с которыми мы работаем, больше не отвечают за операции, которые производятся над ними. Они не обязаны знать о них.
Нет больших иерархий наследования.
Нет указателей (для пользователя)
Нет ручного управления памятью (для пользователя)
Улучшили производительность
И это всё в
private
секции нового классаShape
!
Заключение. Что такое Type Erasure?
Шаблонный конструктор
Полностью НЕвиртуальный интерфейс (спасибо, External Polymorphism)
External Polymorphism + Bridge + Prototype
Очень элегантный современный паттерн :)
Что Type Erasure позволяет сделать?
Избавиться от зависимостей
Избавиться от указателей
Улучшить производительность
Улучшить читабельность и "понимательность" кода
Упростить сопровождение кода
Диаграмма паттерна Type Erasure (в данном примере):
Полный код
#include <iostream>
#include <memory>
#include <vector>
class Shape {
private:
struct ShapeConcept {
virtual void drawCall() const = 0;
virtual std::unique_ptr<ShapeConcept> clone() const = 0;
virtual ~ShapeConcept() = default;
};
template<typename T>
struct ShapeModel : public ShapeConcept {
T shape_instance;
explicit ShapeModel(T &&shape) : shape_instance(std::move(shape))
{}
explicit ShapeModel(const T &shape) : shape_instance(shape) {}
[[nodiscard]] std::unique_ptr<ShapeConcept> clone() const override
{
return std::make_unique<ShapeModel>(*this);
}
void drawCall() const override {
draw(shape_instance);
}
};
std::unique_ptr<ShapeConcept> shapePtr;
public:
template<typename T>
explicit Shape(T &&shape)
: shapePtr(new ShapeModel<T>(std::forward(shape))) {}
friend void draw(const Shape &shape) {
shape.shapePtr->drawCall();
}
Shape(const Shape &other) : shapePtr(other.shapePtr->clone()) {
}
};
class Circle {
public:
explicit Circle(double r)
: radius(r) {}
double getRadius() const {
return radius;
}
void setRadius(double r) {
radius = r;
}
private:
double radius;
};
void draw(const Circle &s) {
std::cout << "I am Circle with radius = " <<
s.getRadius() << std::endl;
}
struct Square {
};
void draw(const Square &s) {
std::cout << "I am Square" << std::endl;
}
int main() {
Shape circle(Circle{3.14});
Shape square(Square{});
// Shape not_supported(123); // не скомпилируется
draw(circle);
draw(square);
std::vector<Shape> v;
for (int i = 0; i < 5; ++i) {
if (rand() % 2 == 0)
v.emplace_back(circle); // конструктор копирования!
else
v.emplace_back(square);
}
for (const auto &shape: v) {
draw(shape);
}
return 0;
}
Источники
Breaking Dependencies: Type Erasure - A Design Analysis - Klaus Iglberger - CppCon 2021
Abstraction Can Make Your Code Worse - про coupling (связность).
CppCon 2017: Nicolai Josuttis “The Nightmare of Move Semantics for Trivial Classes”.(См. конструктор
Shape
)CppCon 2019: Chandler Carruth “There Are No Zero-cost Abstractions” - про производительность умных указателей, да и в целом про производительность разных абстракций.
CppCon 2014: Zach Laine "Pragmatic Type Erasure: Solving OOP Problems w/ Elegant Design Pattern"
Jason Turner. C++ Weekly - Ep 343 - Digging Into Type Erasure
Design Patterns: Elements of Reusable Object-Oriented Software / Erich Gamma, Richard Helm, Ralph Johnson, John Vlissides. - United States : Addison-Wesley, 1994. - 395 c. - ISBN 0-201-63361-2.
Alexander Shvets. Dive Into Design Patterns. - электронная книга : Refactoring.Guru, 2018. - 406 c.
Комментарии (41)
KudryashovDA
19.12.2022 22:29Спасибо за такой подробный разбор паттерна. Казалось бы сложные вещи, но так понятно написано, что прямо хочется взять и попробовать повторить. Жду еще статей про паттерны. Интересно, насколько часто встречаются в реальном продуктовом коде такие решения. Если присутствуют шаблоны - нужно быть очень аккуратным, иначе замучаешься изучать логи ошибок компиляции.
fk0
19.12.2022 22:39+2Полностью НЕвиртуальный интерфейс...
Как не выкручивайтесь, а диспетчеризацию времени исполнения обойти нельзя. Она где-то зарыта. Ну конечно, у вас же виртуальные функци в базовом классе -- вот там и зарыта.
0xd34df00d
19.12.2022 23:58Можно, если вы в итоге статически знаете с достаточной точностью, где и что у вас вызывается.
Условно, если вы имеете код вида
int callFun(std::function<int (int)> f) { return f(42); } int bar() { return callFun([](int n) { return n * 2; }); }
то вы вполне можете обойтись без диспетчеризации.
akuprin Автор
20.12.2022 01:48Насколько я понял пока изучал ассемблерный код примеров на godbolt.org, в этом деле нам очень сильно могут помочь отпимизации компилятора ("девиртуализация", кажется). (Не ручаюсь за свои слова здесь на 100%, я не эксперт ассемблера).
gdt
19.12.2022 22:43+2Спасибо, интересная статья. А почему мы не можем абстрагировать сам рендерер (IVulkanRenderer, IDxRenderer, IOpenGlRenderer), и сделать то же самое с сериализацией? К тому же, composition over inheritance это один из базовых принципов, поэтому примеры выглядят с этой позиции немного надуманно - никто не будет так делать, как вы и сказали.
Kelbon
19.12.2022 22:52+1Не понимаю. Буквально несколько дней назад вы писали в моём посте описывающем собственно type erasure и готовые решения описанных в статье проблем и отвечали, что нужно сделать перегрузку на макросах в С как будто это то же самое(спойлер - нет)
https://habr.com/ru/post/703846/comments/#comment_25020336
А сейчас статью про стирание типов...
https://github.com/kelbon/AnyAny
вот держите блин, реализовано и делает гораздо лучше чем описанное вами стирание типов, с более удобным интерфейсом, убиранием лишних аллокаций и всякими крайне удобными и полезными фишкамиKelbon
19.12.2022 22:58Ваш код это буквально первый пример оттуда
fk0
19.12.2022 23:17Она в момент конструирования any_drawable имеет такой же шаблонный конструктор, как я понимаю, в котором сохраняет указатель на Draw::do_invoke() в себе (условно, скорей там статический класс из шаблона инстанцируется, в нём все указатели, а на него ссылка в объекте) и использует при вызове invoke. Та же таблица виртуальных функций, но сделанная вручную.
Kelbon
19.12.2022 23:18??? Во первых не вручную, а шаблон
Во вторых у вас есть какие то другие варианты как это сделать?
В третьих, в статье описано то же самое, только хужеMingun
20.12.2022 07:14-1Почему хуже, если других вариантов, как это сделать, нет?
Kelbon
20.12.2022 12:30в рамках одного варианта можно написать разный код
Mingun
20.12.2022 17:15Можно, но хуже то он в чем? А то пока выглядит так, что сделал не я — значит гамно.
Kelbon
20.12.2022 17:25в интерфейсе, расширяемости и выделении лишней памяти
Mingun
20.12.2022 17:37-1Ну, память возможно действительно можно не выделять, а что за проблемы с интерфейсом и расширяемостью?
Kelbon
20.12.2022 17:55попробуйте добавить новый тип или новый метод в интерфейс из статьи
Mingun
20.12.2022 18:17+1Так вроде же в последнем листинге написано. Новый тип:
struct NewType {}; void draw(const NewType &s) { std::cout << "I am NewType" << std::endl; }
Про новые методы в самом начале было сказано, что набор операций ограничен и конечен. Не понимаю, зачем требовать от решения того, для чего оно не разрабатывалось.
Кстати, а в вашем варианте как решаются аналогичные вопросы?
Кстати, на вопрос, в чем проблемы интерфейса решения (и что вы под этим подразумеваете) вы так и не ответили.
Kelbon
20.12.2022 19:10+1В случае добавления нового типа с нужным интерфейсом всё тривиально - просто пишете этот тип.
Если нужно добавить метод в интерфейс, вы просто пишете этот метод и вставляете в собственно интерфейс, например
template<typename T> struct Foo { ... }; using any_xxx = any_with<СтарыеМетоды..., Foo>;
Решение из статьи уже в этом моменте ломается, нужно менять реализацию на уровне ShapeConcept и проч.
В этот момент метод Foo начинает требоваться для всех типов, где использовано any_xxx.
Но можно сделать интереснее.
Если у вас раньше функция требовалаint bar(any_with<OldMethods...>::ref x);
То теперь изменения сигнатуры не нужны. Потому что any_with<OldMethods..., Foo>::ref неявно приводится к any_with<OldMethods...>
Таким образом типам отправленным в функцию bar не требуется реализовывать новый метод из интерфейса. ABI и API брейка нет.
Этого не добиться ни с каким решением с монолитным интерфейсом или любым тайп ерайзом использущим в реализации virutal.
Вы можете требовать в сигнатуре функции только те методы, которые ей правда нужны - этого опять же не добиться ни с виртуальными функциями ни с решением из статьи.
Методы можно переиспользовать в других инерфейсах : как бы собирать из готовых блоков новый тип.
При создании any_with<...>::ref или any_with<...>::ptr не происходит никакого выделения памяти вовсе, компилятор отлично это оптимизирует(лучше, чем виртуальные функции)
При создании самого объекта any_with<...> тоже скорее всего аллокация не будет сделана, потому что там оптимизация аналогичная SSO в строках
В общем это решение таково, что изменения кода с ним минимальны насколько это теоретически возможно при расширении функционала + переиспользование статического полиморфизма и возможность убрать управление памятью и virtual из кода вовсе. Ну остальное(что можно делать с этими типами, например мультидиспетчеризацию) смотрите в ридми на гитхабе
fk0
19.12.2022 23:12+1_Generic никакого отношения к макросам не умеет. Это такой switch-case времени компиляции, который позволяет выбрать выражение в зависимости от типа. С ним проблема в том, что все типы нужно заранее перечислить. В чистом виде, как задумывалась -- достаточно бесполезная штука (а задумывалась наверняка из-за ряда библиотечных функций, вроде атомиков, которые бы иначе имели совсем сумасшедший интерфейс).
На основе _Generic становятся возможными некоторые трюки, что гораздо более полезны, чем собственно switch-case времени компиляции (который всегда был в виде __builtin_choose_expr()). Например, можно отличать константные и не константные указатели, можно отличать константы времени компиляции от переменных. Можно вообще как-то программировать компилятор (собственно это единственный оператор выбора доступный в стандартном ISO C в момент компиляции). Но для реализации полиморфных функций, _Generic -- достаточно бесполезная сущность, повторюсь, из-за необходимости знания всех типов наперёд и постоянного их перечисления.
akuprin Автор
20.12.2022 03:26Да, я уже тогда понял, что есть разница, а я сам написал о другом, спасибо! И ваша библиотека, которая делает такое в рантайме, тоже отличная.
maeris
19.12.2022 23:21+1У вас по-прежнему используется динамическая диспетчеризация.
Задача расширения кода на новые типы объектов и новые методы без нелокальных изменений называется expression problem. Её 15 лет назад уже решили, почитать можно, например здесь. На С++ это будет выглядеть как CRDT + инстанцирования классов с статическими методами.
vamireh
20.12.2022 12:33На С++ это будет выглядеть как CRDT + инстанцирования классов с статическими методами
Можно пример?
Deosis
20.12.2022 07:55+1Статья хорошая, но почему перепутаны паттерны стратегия и визитор?
В стратегии часть алгоритма выносится в отдельный класс. Например, функция сравнения в алгоритме сортировки, либо функция хеширования в ассоциативном контейнере.
А визитор вызывает отдельный алгоритм в зависимости от типа класса.
Пример реализации паттерна
struct ShapeVisitor; struct Shape { void Accept(const ShapeVisitor&) = 0; }; struct Circle; struct Square; struct ShapeVisitor { void Visit(const Circle&) = 0; void Visit(const Square&) = 0; }; struct Square : public Shape { void Accept(const ShapeVisitor& v) final { v.Visit(*this); } } struct DrawVisitor : public ShapeVisitor { void Visit(const Circle& c) final { ... } }; struct SerializeVisitor : public ShapeVisitor { void Visit(const Circle& c) final { ... } };
akuprin Автор
21.12.2022 02:30А в моём примере из класса фигуры выносится алгоритм её рисования. Считаю, что это всё-таки стратегия. Но прочитав ваш пример понял, что можно решить проблему и с помощью паттерна посетитель. Оба они в рамках данной задачи имеют место быть, просто говорить о них, считаю, нужно в разных местах.
Если идти по тем же шагам, что в статье. На тот момент, когда приводится решение проблемы с помощью стратегии, мы ещё не отказались от наличия метода
draw
в классах фигур.Шаблон стратегия позволяет переключаться между алгоритмами (стратегиями) в зависимости от ситуации. Вот мы и пользуемся этим паттерном:
struct Circle : public Shape { std::unique_ptr<DrawStrategy> drawStrategy; explicit Circle(DrawStrategy *drawStrategy) : drawStrategy(drawStrategy) { } void draw() override { drawStrategy->draw(this); } };
Таким образом мы выделили в отдельный класс метод рисования.
С другой стороны: Шаблон посетитель позволяет добавлять будущие операции для объектов без их модифицирования. Вот на этапе, когда мы дошли до того, что в классах фигур больше нет метода
draw
(несколькими шагами позже примера со стратегией), думаю можно уже реализовывать визитора. Именно на этом моменте мы перестаём модифицировать классы, с которыми работаем, прямо как в кратком определении визитора, которое я привёл в начале абзаца.И главное - мы их больше никогда не изменим!
В этот момент начинается рассказ о паттерне external polymorphism, поэтому визитор слегка отходит на второй план.
kosmonaFFFt
21.12.2022 11:02+1Я примерно 10 лет назад с таким упражнялся вот тут: https://habr.com/ru/post/151504/
Делал нечто похожее, но с помощью std::function. Никакой практической ценности я тогда из этого не извлек, но как упражнение для мозгов сгодилось.
izvolov
Так и не понял, в чём элегантность и "расширяемость" по сравнению, к примеру, с вариантом:
Как говорится, нет такого преступления, на которое не пойдут джависты, даже под страхом виселицы, лишь бы обмазаться 300% паттернов.
izvolov
А за виртуальный метод
clone
в плюсах полагается расстрел на месте.izvolov
Как делается настоящее стирание типа без виртуальной шляпы — см.
std::function
.Более продвинуто — Boost.TypeErasure.
sergegers
Да я регулярно здесь читаю статьи таких вот клоунов. Невооружённым глазом видно, что человек не имеет опыта программирования на плюсах, но уже учит других.
vkni
Из говномёта! Прямо под стенами здания вместе с группой коллег, пропустивших на рецензии! :-)
akuprin Автор
Почему? Объясните, пожалуйста.
fk0
А если все возможные объекты наперёд заранее не известны? Т.е. в момент линковки-то они конечно известны, но сделать так, чтоб был такой вариант который знает все типы -- не вариант.
Mingun
Автору следовало уточнить, что паттерн имеет смысл, когда создание и использование
Shape
делается кодом разных владельцев (скажем, создание в плагине, а вызовdraw
— в плагин-хосте). Иначе и городить ничего не надо, достаточноakuprin Автор
Согласен. А то, что вы говорите, слегка упомянуто мной:
См. ссылка
akuprin Автор
izvolov
std::function, Boost.TypeErasure, Boost.Polycollection
akuprin Автор
Как минимум, код становится красивее и читабельнее! А с
std::variant
могут возникнуть трудности, например: https://godbolt.org/z/633eh11rc