SOLID Principle
Single Responsibility Principle (SRP) 日记例子,一个类只做一件事
Open-Closed Principle (OCP)
根据颜色大小排序, Specification类,为扩展开发,为修改关闭
Liskov Substitution Principle (LSP)
父类定义的方法意义,子类不能相违背,矩形和正方形的set_size()
Interface Segregation Principle (ISP)
打印机传真机扫描仪封为machine,不如打印,传真,扫描封为抽象功能组合
Dependency Inversion Principle (DIP)
其他类依赖本类的高级抽象而非具体实现,类间的依赖原则
Creational Patterns
创建型模式
一些概念
- 栈分配
- 堆分配
- 智能指针
Builder
创建模式
Simple Builder
HtmlBuilder
Fluent Builder
使用流式操作
.add_child("li", "hello").add_child("li", "world");
builder->add_child("li", "hello")->add_child("li", "world"); builder
为了强制让人使用Builder而不是构造函数, 可以把构造函数设为protected类型
类的隐式转换
#include <cstdio>
class Html
{
public:
void str()
{
("str");
printf}
protected:
() = default;
Htmlfriend class HtmlBuilder;
};
class HtmlBuilder
{
;
Html html
public:
operator Html() const
{
return html;
}
};
int main()
{
;
HtmlBuilder builder= builder;
Html root .str();
root}
Groovy-Style Builder
Tag抽象类, 包含了所有的数据存储, 构造函数protected
P, IMG等继承, 构造函数public, 初始化一些值
总结: 通过派生类特定构造来实现多样化的对象生成
import groovy.transform.builder.Builder
Builder(builderMethodName = 'initiator', buildMethodName = 'create')
@class Message {
String from, to, subject, body
}
def message = Message.initiator()
.from('mrhaki@mrhaki.com')
.body('Groovy rocks!')
.create()
assert message.body == 'Groovy rocks!'
assert message.from == 'mrhaki@mrhaki.com'
mini-DSL
例如下面Swift中的VStack
、Text
和Button
都是SwiftUI嵌入在Swift中的DSL
var body: some View {
{
VStack ("Hello, World!")
Text.font(.largeTitle)
("Tap me!") {
Button("Button was tapped")
print}
}
}
Composite Builder
Person静态函数create()生成PersonBuilder, PersonBuilder利用.at(), .with_post_code(), .as_a()等fluent builder快速构建对象, 然后利用operator Person转化为Person对象
Factories
工厂模式
Factory Method
类的方法替代构造函数创建类
Point, 笛卡尔坐标, 极坐标, 静态NewCartesian, NewPolar方法, 构造函数protected类型
Factory
隔离的类创建对象, 即使只是提供函数来创建类而不是直接创建
把静态NewCartesian, NewPolar方法放到PointFactory里面, Point友元类为工厂类, Point构造函数去掉
Inner Factory
把工厂类设置为Point的内置类, 可以访问Point私有成员而不需要友元, 然后声明一个静态的工厂类对象在Point类中
Abstract Factory
能被实际工厂函数继承
要创建的对象继承关系多种类复杂, 比如drink, hotdrink, tea, coffee, 工厂函数也要有相应的继承关系
和Builder的区别, Builder是一点数据一点数据添加, Factory是一次性生成
Prototype
原型模式, 对象模型能够直接拷贝, 然后通过修改参数生成新的对象
Object Constrution
构造两个类似的对象通常需要花费更多操作, 拷贝则简单许多
Ordinary Duplication
直接用等于号复制, 会带来浅拷贝的问题
Duplication via Copy Construction
用拷贝构造函数来拷贝对象或者用=操作符拷贝对象, 或者继承自Clonable类用clone方法, 用CRTP
Serialization
使用Boost的Serialization, 侵入式, 非侵入式, 原理是使用序列化和反序列化来传递数据生成新的对象
Prototype Factory
预设几个全局的模板, 用工厂方法生成对应的
Singleton
单例模式
Singleton as Global Object
静态变量返回
Classic Implementation
构造函数加计数器, 禁用各种构造函数, 删除拷贝构造, 赋值操作符
Thread Safety
加锁防止同时调用初始化函数造成问题
The Trouble with Singleton
单例模式写死后测试需要用假的类, 比如database, 需要改动就有很复杂的依赖, 尝试依赖单例对象的抽象接口类而不是依赖单例类
Singletons and Inversion of Control
使用IoC模式减少单例模式中单例对象的依赖性, 用依赖注入DI
Monostate
单态
是个普通的类但是表现为单例, 可以进行继承实现多态
Structural Patterns
结构型模式
几种结构:
- Inheritance: 继承
- Composition: 包含(拥有对象)
- Aggregation: 聚集(拥有对象指针)
Adapter
适配器模式,STL中的容器适配器和迭代器适配器就是适配器模式的应用之一
Adapter
画点函数需要画所有线的点, 包括线组成的对象, 需要把线转成点集然后绘制
Adapter Temporaries
上述操作每次都要进行一次转换, 消耗计算资源, 利用哈希存储转换后的数据可以只转换不存在的点
类适配器模式
类适配器模式是适配器模式的一种,其定义为将一个类的接口转换成客户端所期望的另一个接口,使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。在类适配器模式中,适配器与适配者之间是继承关系。
// 接口
class Target {
public:
virtual void request() = 0;
};
// 与接口不兼容的类
class Adaptee {
public:
void specificRequest() {
std::cout << "Adaptee::specificRequest()" << std::endl;
}
};
// 通过多继承的方式实现
class Adapter : public Adaptee, public Target {
public:
void request() override {
();
specificRequest}
};
int main() {
* target = new Adapter();
Target->request();
target
delete target;
return 0;
}
对象适配器模式
相对于类适配器模式,对象适配器模式使用组合的方式,而不是继承的方式来实现适配器
class Target {
public:
virtual void request() = 0;
};
class Adaptee {
public:
void specificRequest() {
std::cout << "Adaptee::specificRequest()" << std::endl;
}
};
class Adapter : public Target {
public:
(Adaptee* adaptee) : m_adaptee(adaptee) {}
Adapter
void request() override {
m_adaptee->specificRequest();
}
private:
* m_adaptee;
Adaptee};
int main() {
* adaptee = new Adaptee();
Adaptee* target = new Adapter(adaptee);
Target->request();
target
delete target;
delete adaptee;
return 0;
}
接口适配器
接口适配器模式用于解决一个接口有多个方法,而客户端只需要使用其中的部分方法的情况。在接口适配器模式中,适配器通过实现一个或多个接口来适配适配者与客户端之间的接口
class Target {
public:
virtual void request() = 0;
};
class Adaptee {
public:
void method1() {
std::cout << "Adaptee::method1()" << std::endl;
}
void method2() {
std::cout << "Adaptee::method2()" << std::endl;
}
void method3() {
std::cout << "Adaptee::method3()" << std::endl;
}
};
// 使用Adapter类只实现method1接口
class Adapter : public Target, public Adaptee {
public:
void request() override {
();
method1}
};
int main() {
* target = new Adapter();
Target->request();
target
delete target;
return 0;
}
Bridge
桥接模式,用于将抽象部分与它的实现部分分离,使它们都可以独立地变化
The Pimpl Idiom
Pimpl的好处:
- 只暴露public接口
- 修改私有类的时候由于是指针存在公有类中, 不会对公有类的二进制结构造成影响, 比如size或者函数相对偏移地址等
- 减少相互包含更多的头文件, 私有类的包含可以放到cpp文件里面去, 头文件不需要改不会让其他包含该头文件的文件重新编译
Example
Shape的自绘, Renderer的种类, Circle继承自形状, 传入Renderer引用存储
// 定义 Shape 类,作为抽象化角色。它持有一个 Renderer 类型的指针,并在 draw 函数中调用 Renderer 的 renderCircle 函数。
class Renderer {
public:
virtual void renderCircle(float x, float y, float radius) = 0;
};
class Shape {
protected:
* renderer;
Renderer
public:
(Renderer* renderer) : renderer(renderer) {}
Shape
virtual void draw() = 0;
virtual void resize(float factor) = 0;
};
// 定义 Circle 类,作为改善后的抽象化角色。它继承自 Shape 类,增加了一个 radius 成员变量,并在 draw 函数中调用 Renderer 的 renderCircle 函数。
class Circle : public Shape {
private:
float x, y, radius;
public:
(Renderer* renderer, float x, float y, float radius)
Circle: Shape(renderer), x(x), y(y), radius(radius) {}
void draw() override {
->renderCircle(x, y, radius);
renderer}
void resize(float factor) override {
*= factor;
radius }
};
// 定义 Renderer 类及其两个具体实现类 RedCircle 和 GreenCircle,作为实现者角色。它们都实现了 renderCircle 函数,用于渲染圆形。
class Renderer {
public:
virtual void renderCircle(float x, float y, float radius) = 0;
};
class RedCircle : public Renderer {
public:
void renderCircle(float x, float y, float radius) override {
<< "Drawing red circle at (" << x << ", " << y << ") with radius " << radius << endl;
cout }
};
class GreenCircle : public Renderer {
public:
void renderCircle(float x, float y, float radius) override {
<< "Drawing green circle at (" << x << ", " << y << ") with radius " << radius << endl;
cout }
};
// 绘制
int main() {
* renderer = new RedCircle();
Renderer* circle = new Circle(renderer, 5, 5, 10);
Shape->draw();
circle
->resize(2);
circle->draw();
circle
delete circle;
delete renderer;
= new GreenCircle();
renderer = new Circle(renderer, 10, 10, 20);
circle ->draw();
circle
delete circle;
delete renderer;
return 0;
}
使用桥接模式可以将抽象和实现部分分离,从而使得它们可以独立变化。这样,我们就可以轻松地实现不同形状和不同颜色之间的组合,并且可以在运行时动态地改变它们之间的关系
Composite
组合模式使得用户对单个对象和组合对象的使用具有一致性
给单一的多个对象或者一组对象同一的接口, 比如begin/end, 单个的Graph对象有draw, 多个Graph组成的对象组同样有draw
ArcGIS Qt SDK里面的Geometry也是类似的设计方式, 包括了Point, Polyline, Polygon, Multipart等等
Array Backed Properties
Creature有一些值比如力量, 敏捷和智力, 要统计这些数据的共性, 与其定义几个不同的变量不如用数组和枚举记录在一起, 用迭代器求均值最大最小值
Grouping Graphic Objects
针对相同接口的对象进行编组, 用虚函数调用
Neural Networks
派生自模板接口
template <typename Self>
struct SomeNeurons
{
template <typename T> void connect_to(T& other)
{
for (Neuron& from : *static_cast<Self*>(this))
{
for (Neuron& to : other)
{
.out.push_back(&to);
from.in.push_back(&from);
to}
}
}
};
struct Neuron
{
<Neuron*> in, out;
vectorunsigned int id;
()
Neuron{
static int id = 1;
this->id = id++;
}
}
struct SingleNeuron : Neuron,
<SingleNeuron>
SomeNeurons{
}
struct NeuronLayer : Neuron,
<Neuron>,
vector<NeuronLayer>
SomeNeurons{
}
Decorator
装饰模式
在不改变原有类的情况下增强类的功能并且把增强的东西隔离开
实现方法:
- 动态的组合允许在运行时组合, 拥有最大的弹性
- 静态的组合对象的增强是在编译时使用模板, 所以在编译时就需要知道所有可能的增强
Dynamic Decorator
通过子类传入父类的引用或者指针来增加功能, 多个功能通过构造函数嵌套实现多种功能, 通过多态
struct ColoredShape : Shape
{
& shape;
Shape;
string color(Shape& shape, const string& color) :
ColoredShape{shape}, color{color} {}
shape};
struct TransparentShape : Shape
{
& shape;
Shapeuint8_t transparency;
(Shape& shape, uint8_t transparency) :
TransparentShape{shape}, transparency{transparency} {}
shape};
// 多个功能的, 使用多态实现功能的叠加
{ColoredShape{Circle{1.2}, "red"}, 100}; TransparentShape s
Static Decorator
使用Mixin方式继承模板类, 添加功能增强类和子类
template <typename... Mixins>
class Point : public Mixins... {
public:
double x, y;
() : Mixins()..., x(0.0), y(0.0) {}
Point(double x, double y) : Mixins()..., x(x), y(y) {}
Point};
class Label {
public:
std::string label;
() : label("") {}
Label};
class Color {
public:
unsigned char red = 0, green = 0, blue = 0;
};
using MyPoint = Point<Label, Color>;
该例子实现了带颜色和标签的点,功能组合了
Function Decorator
使用函数模板为入参, 在函数开始前和结束后进行附加操作, 增强函数的功能, 比如记录函数运行时间和运行状态的Logger类, 重载()运算符
例如python的函数装饰器
import logging
import time
def logger(func):
='example.log', level=logging.INFO)
logging.basicConfig(filename
def wrapper(*args, **kwargs):
= time.time()
start_time = func(*args, **kwargs)
result = time.time()
end_time = end_time - start_time
run_time "Function {}() ran in {} seconds with result: {}".format(
logging.info(__name__, run_time, result))
func.return result
return wrapper
@logger
def add(x, y):
return x + y
print(add(2, 3)) # 输出 5,同时将日志写入 example.log 文件中
Façade
外观模式, 终端就是外观, 实际的内部形式是buffer, 把简单接口放在复杂的子系统前面, 比如终端很多的buffer和viewport, 最后抽象成API。UE5里面的MeadiaPlayer也用到了外观模式。
外观模式的优点在于它能够简化客户端代码,降低客户端与子系统之间的耦合度。同时,外观模式还能够提高代码的可维护性和可扩展性,因为它将子系统的复杂性封装在一个单独的类中,使得对子系统的修改和扩展更加容易
#include <iostream>
// 子系统类A
class SubsystemA {
public:
void operationA() {
std::cout << "Subsystem A operation" << std::endl;
}
};
// 子系统类B
class SubsystemB {
public:
void operationB() {
std::cout << "Subsystem B operation" << std::endl;
}
};
// 外观类
class Facade {
private:
* subsystemA;
SubsystemA* subsystemB;
SubsystemBpublic:
() {
Facade= new SubsystemA();
subsystemA = new SubsystemB();
subsystemB }
~Facade() {
delete subsystemA;
delete subsystemB;
}
// 对外提供的简单接口
void operation() {
->operationA();
subsystemA->operationB();
subsystemB}
};
// 客户端代码
int main() {
* facade = new Facade();
Facade->operation();
facadedelete facade;
return 0;
}
Flyweight
享元模式
(flyweight token or cookie) 通常是一个临时组件, 比如智能指针, 通常有大量相同的对象, 需要最小化使用内存存储。例如字符串表哈希等。
User Names
大型服务器相同名字的用户每个用户都存储一份字符串会消耗大量的字节
用int代表key, 去bitmap中查找名称, bitmap可用双向查找
Boost.Flyweight
boost::flyweight<string>直接可用变成享元
String Ranges
std::string::substring()
返回的是额外的字符串空间,
而不是共享内存只改变下标的字符串
string_view
则是共享单元
Naïve Approach
建立一个bool数组对应string每个字符, bool为true代表要把对应字符串大写, 这样一个字符串既存储了原始字符串也存储了大写后的字符串
vector<bool>是优化存储过的, 用位
Flyweight Implementation
格式化字符串, 用和上面一样的方式存储多个TextRange类, TextRange里面标记起始和结束, 代表一个格式
享元模式没有确定的形式, 有时候是API token, 有时候是隐式的, 藏在幕布后面, 客户端并不知道是否使用享元模式
Proxy
装饰模式是不同方式增强对象的功能, 代理模式相似, 但是目标是在提供增强功能时尽可能保持正在使用的API
Proxy不是同质的, 不同类型的人建立方式不同, 服务目标不同
Smart Pointers
智能指针在实现其额外功能时保留了和普通指针一样的特性接口, 比如*运算符和->运算符, 还有bool运算符, 下标运算符, 但是智能指针不能执行delete 操作
Property Proxy
实现了getter, setter和普通的值一样
Virtual Proxy
lazy instantiation(延迟实例化), 仅用到了的时候才对对象构造分配空间, eager(反义词)
LazyBitmap, 只有在调用draw的时候才进行图片加载, 接口一样但是实现的方式不一样
Communication Proxy
本机通信和跨计算机通信共用接口
总结:
不像装饰模式, 代理模式不会通过增加对象成员来扩大功能(除非没办法). 所有对已知功能的增强都是基于已经存在的成员
- property proxies是替身对象, 替换fields同时在获取值和写入值时进行额外操作
- virtual proxies提供虚拟的接触基础对象, 然后实现例如lazy loading之类的功能. 这样让你感觉跟和真正的对象接触一样.
- communication proxies允许改变对象的物理位置, 比如移动到云端, 但是我们只需要使用相同的API
- logging proxies允许我们在调用基础函数的时候记录日志, 参考Function Decorator. 区别在于用代理模式实现的话两个函数同名
Behavioral Patterns
行为型模式, 没有一个固定的主题, 只有相似的策略, 大多数模式以独特的表现形式解决特定的问题
Chain of Responsibility
职责链模式
Pointer Chain
class CreatureModifier
{
* next{nullptr};
CreatureModifierprotected:
& creature; // alternative: pointer or shared_ptr
Creaturepublic:
explicit CreatureModifier(Creature& creature):
(creature) {}
creature
void add(CreatureModifier* cm)
{
if (next) next->add(cm);
else next = cm;
}
virtual void handle()
{
if (next) next->handle(); // critical!
}
};
class DoubleAttackModifier : public CreatureModifier
{
public:
explicit DoubleAttackModifier(Creature &creature)
: CreatureModifier(creature) {}
void handle() override
{
.attack *= 2;
creature::handle();
CreatureModifier}
};
类似的QWidget的paintEvent也是递归调用。组合模式加职责链模式可以实现集合中所有元素遍历操作
Broker Chain
通过一个集中化组件, 该组件保存一个Modifier列表,Broker理解为中间件
Boost.Signals2实现信号槽
Command Query Separation (CQS)
Gang of Four(GoF,Erich Gamma, Richard Helm, Ralph Johnson和John Vlissides)
struct Game // mediator
{
<void(Query & )> queries;
signal};
struct Query
{
;
string creature_nameenum Argument
{
, defense
attack} argument;
int result;
};
class Creature
{
&game;
Game int attack, defense;
public:
string name
(Game &game, ...) : game{game}, ...
Creature{ ... }
// other members here
};
int Creature::get_attack() const
{
{name, Query::Argument::attack, attack};
Query q.queries(q);
gamereturn q.result;
}
class CreatureModifier
{
&game;
Game &creature;
Creature public:
(Game &game, Creature &creature)
CreatureModifier: game(game), creature(creature)
{}
};
class DoubleAttackModifier : public CreatureModifier
{
;
connection connpublic:
(Game &game, Creature &creature)
DoubleAttackModifier: CreatureModifier(game, creature)
{
= game.queries.connect([&](Query &q) {
conn if (q.creature_name == creature.name &&
.argument == Query::Argument::attack)
q.result *= 2;
q});
}
~DoubleAttackModifier()
{ conn.disconnect(); }
};
;
Game game{game, "Strong Goblin", 2, 2};
Creature goblin<< goblin << endl;
cout // name: Strong Goblin attack: 2 defense: 2
{game, goblin};
DoubleAttackModifier dam<< goblin << endl;
cout // name: Strong Goblin attack: 4 defense: 2
<< goblin << endl;
cout // name: Strong Goblin attack: 2 defense: 2
Command
命令模式是一种行为型设计模式,它将一个请求封装为一个对象,从而允许我们使用不同的请求对客户端进行参数化。请求被包装在对象中,并传递给调用对象,调用对象寻找可以处理该命令的合适对象,并将该命令传递给相应的对象,该对象执行命令。
给一个值赋值, 让其改变, 并不能知道改变记录, 也没办法回滚. 命令模式与其通过API改变对象, 不如发送命令: 指导该怎样做
#include <iostream>
using namespace std;
// 命令接口
class Command {
public:
virtual void execute() = 0;
};
// 具体命令
class ConcreteCommand : public Command {
public:
() {}
ConcreteCommand~ConcreteCommand() {}
void execute() {
<< "执行命令..." << endl;
cout }
};
// 调用对象
class Invoker {
public:
(Command *cmd) {
Invokerm_cmd = cmd;
}
~Invoker() {
delete m_cmd;
m_cmd = nullptr;
}
void invoke() {
m_cmd->execute();
}
private:
*m_cmd;
Command };
// 客户端
int main() {
*cmd = new ConcreteCommand();
Command *invoker = new Invoker(cmd);
Invoker ->invoke();
invoker
delete invoker;
return 0;
}
Implementing the Command Pattern
定义虚基类Command带纯虚函数call, BankAccountCommand构造函数带BankAccount, Action, amout, 调用call时存钱取钱
Undo Operations
Command添加undo虚函数
#include <iostream>
#include <vector>
using namespace std;
// 命令接口
class Command {
public:
virtual void execute() = 0;
virtual void undo() = 0;
};
// 具体命令类
class ConcreteCommand : public Command {
public:
(int& value, int delta) : m_value(value), m_delta(delta) {}
ConcreteCommandvirtual void execute() override {
m_prevValue = m_value;
m_value += m_delta;
}
virtual void undo() override {
m_value = m_prevValue;
}
private:
int& m_value;
int m_delta;
int m_prevValue;
};
// 命令管理类
class CommandManager {
public:
void execute(Command* cmd) {
m_commands.push_back(cmd);
->execute();
cmd}
void undo() {
if (!m_commands.empty()) {
* cmd = m_commands.back();
Commandm_commands.pop_back();
->undo();
cmddelete cmd;
}
}
private:
<Command*> m_commands;
vector};
int main() {
int value = 0;
;
CommandManager manager.execute(new ConcreteCommand(value, 1)); // value = 1
manager.execute(new ConcreteCommand(value, 2)); // value = 3
manager.execute(new ConcreteCommand(value, 3)); // value = 6
manager.undo(); // value = 3
manager.undo(); // value = 1
manager.undo(); // value = 0
manager.undo(); // do nothing
managerreturn 0;
}
Composite Command
组合命令, 转账, 提款后取款, 组合模式
struct CompositeBankAccountCommand : vector<BankAccountCommand>, Command
组合命令需要全部执行才能算成功, 所以call和undo需要判断, 中间步骤出错也需要把已经做的步骤撤回等等
Command Query Separation
命令查询分开
- Commands, 指导系统执行一些操作参与状态变化但是没有返回值
- Queries, 返回值但是不改变状态
class Creature
{
int strength, agility;
public:
(int strength, int agility)
Creature: strength{strength}, agility{agility}
{}
void process_command(const CreatureCommand &cc);
int process_query(const CreatureQuery &q) const;
};
Interpreter
解释器模式
显著的例子:
- 数字字面量, 需要用二进制存储, 转换函数atof
- 正则表达式, 能够在字符串中匹配特定的模式, (domain-specific-language)DSL(领域专用语言), 使用之前必须保证正确解释
- 结构化的数据, 比如csv, xml, json需要先解释后使用
- 成熟的编程语言, 比如c, python在编译或者解释的时候先要对代码进行解释
Numeric Expression Evaluator
数学表达式求值
Lexing
词法分析, 把字符序列转化为tokens符号, tokens是句法的主要元素, 最后以一串平面序列结束, token可以是以下:
- 整数
- 操作符
- 开闭括号
Parsing
语法分析, 把tokens序列转换为有意义的, 结构化的, 面向对象的数据
完整代码:
#include <string>
#include <utility>
#include <iostream>
#include <ostream>
#include <sstream>
#include <vector>
using namespace std;
struct Token
{
enum Type
{
,
t_integer,
t_plus,
t_minus,
t_lparen,
t_rparen};
;
Type type;
string text
explicit Token(Type type, string text) :
(type), text(std::move(text))
type{}
friend ostream &operator<<(ostream &os, const Token &obj)
{
return os << "`" << obj.text << "`";
}
};
<Token> lex(const string &input)
vector{
<Token> result;
vector;
ostringstream bufferauto add_integer = [&] {
if (buffer.precision()) {
.emplace_back(Token{Token::t_integer, buffer.str()});
result.str("");
buffer}
};
for (char c : input) {
switch (c) {
case '+':
();
add_integer.emplace_back(Token{Token::t_plus, "+"});
resultbreak;
case '-':
();
add_integer.emplace_back(Token{Token::t_minus, "-"});
resultbreak;
case '(':
();
add_integer.emplace_back(Token{Token::t_lparen, "("});
resultbreak;
case ')':
();
add_integer.emplace_back(Token{Token::t_rparen, ")"});
resultbreak;
case '0':
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
case '9':
<< c;
buffer break;
default:
break;
}
}
();
add_integerreturn result;
}
struct Element
{
virtual int eval() const = 0;
};
struct Integer : Element
{
int value;
explicit Integer(const int value) :
(value)
value{}
int eval() const override
{ return value; }
};
struct BinaryOperation : Element
{
enum Type
{
,
t_addition,
t_subtraction};
;
Type type<Element> lhs, rhs;
shared_ptr
int eval() const override
{
if (type == t_addition) {
return lhs->eval() + rhs->eval();
} else {
return lhs->eval() - rhs->eval();
}
}
};
<Element> parse(const vector<Token> &tokens)
shared_ptr{
auto result = make_unique<BinaryOperation>();
bool have_lhs = false;
for (int i = 0; i < tokens.size(); ++i) {
auto token = tokens[i];
switch (token.type) {
case Token::t_integer: {
int value = stoi(token.text);
auto integer = make_shared<Integer>(value);
if (!have_lhs) {
->lhs = integer;
result= true;
have_lhs } else {
->rhs = integer;
result}
}
case Token::t_plus:
->type = BinaryOperation::t_addition;
resultbreak;
case Token::t_minus:
->type = BinaryOperation::t_subtraction;
resultbreak;
case Token::t_lparen: {
int j = i;
for (; j < tokens.size(); ++j) {
if (tokens[j].type == Token::t_rparen) {
break;
}
}
<Token> subexpression(&tokens[i + 1], &tokens[j]);
vectorauto element = parse(subexpression);
if (!have_lhs) {
->lhs = element;
result= true;
have_lhs } else {
->rhs = element;
result= j;
i }
}
case Token::t_rparen:
break;
}
}
return result;
}
template<typename T>
&operator<<(ostream &os, vector<T> vec)
ostream {
for (auto &e : vec) {
<< e;
os }
return os;
}
int main()
{
auto tokens = lex("4+(13+2)");
<< tokens << endl;
cout auto parsed = parse(tokens);
<< parsed->eval() << endl;
cout }
Parsing with Boost.Spirit
Boost.Spirit提供简明的API来创建parser, 没有明显区分lexing和parsing步骤, 允许自己定义文本的结构怎样映射成对象类型
Abstract Syntax Tree
抽象语法树
Parser
BNF语法
Printer
Iterator
迭代器模式, 很多数据结构都涉及到遍历
Iterators in the Standard Library
注: range-based for循环需要在类里面实现迭代器, 原生实现方式
template<typename T, int N>
class rray
{
public:
using iterator = T *;
()
iterator begin{
return &data[0];
}
()
iterator end{
return &data[N];
}
&operator[](int index)
T {
return data[index];
}
private:
[N];
T data};
int main(int argc, char *argv[])
{
<int, 10> a{};
Arayfor (auto &it : a) {
= 1;
it }
}
vector里面的迭代器样式: iterator, const_iterator, 函数begin(), cbegin(), rbegin(), crbegin() c代表const, r代表reserved, reserved能够遍历vector中没写数据的部分
Traversing a Binary Tree
pre-oder
遍历二叉树, 前序中序后序
Iteration with Coroutines
post-oder, 用递归的方式
Mediator
中间者模式, 大多数类和类的通信关系都是通过指针引用进行, 也有设计需要两个对象互相不知道对方存在
中间者模式目的是减轻组件之间的通信
#include <iostream>
#include <string>
#include <vector>
using namespace std;
class Colleague;
// 抽象中介者
class Mediator {
public:
virtual void send(string message, Colleague* colleague) = 0;
};
// 具体中介者
class ConcreteMediator : public Mediator {
public:
void setColleague(vector<Colleague*> colleagues) {
this->colleagues = colleagues;
}
void send(string message, Colleague* colleague) {
for (auto c : colleagues) {
if (c != colleague) {
->receive(message);
c}
}
}
private:
<Colleague*> colleagues;
vector};
// 抽象同事类
class Colleague {
public:
(Mediator* mediator) : mediator(mediator) {}
Colleague
virtual void send(string message) = 0;
virtual void receive(string message) = 0;
protected:
* mediator;
Mediator};
// 具体同事类
class ConcreteColleagueA : public Colleague {
public:
(Mediator* mediator) : Colleague(mediator) {}
ConcreteColleagueA
void send(string message) {
->send(message, this);
mediator}
void receive(string message) {
<< "ConcreteColleagueA received message: " << message << endl;
cout }
};
class ConcreteColleagueB : public Colleague {
public:
(Mediator* mediator) : Colleague(mediator) {}
ConcreteColleagueB
void send(string message) {
->send(message, this);
mediator}
void receive(string message) {
<< "ConcreteColleagueB received message: " << message << endl;
cout }
};
int main() {
* mediator = new ConcreteMediator();
ConcreteMediator* colleagueA = new ConcreteColleagueA(mediator);
ConcreteColleagueA* colleagueB = new ConcreteColleagueB(mediator);
ConcreteColleagueB<Colleague*> colleagues = {colleagueA, colleagueB};
vector->setColleague(colleagues);
mediator
->send("Hello, colleagueB!");
colleagueA->send("Hi, colleagueA!");
colleagueB
delete colleagueA;
delete colleagueB;
delete mediator;
return 0;
}
Chat Room
Person接收发送广播, ChatRoom加入人,发送广播, 转发信息给人
Mediator with Events
简单的用类实现类似Observer观察者模式, 另一种办法是用事件, 参与者订阅事件来接收通知, 也可以通过发送消息来触发事件
基础的C++不支持事件, Boost.Signal2提供需要的功能. 其中的术语: signals代表生成通知的对象, slots处理通知的函数
#include <string>
#include <iostream>
#include <boost/signals2.hpp>
using namespace std;
using namespace boost;
struct EventData
{
virtual ~EventData() = default;
virtual void print() const = 0;
};
struct PlayerScoreData : EventData
{
;
string player_nameint goals_scored_so_far;
(string player_name, const int goals_scored_so_far) :
PlayerScoreData(std::move(player_name)), goals_scored_so_far(goals_scored_so_far)
player_name{}
void print() const override
{
<< player_name << "has scored! (their "
cout << goals_scored_so_far << "goal)" << "\n";
}
};
struct Game
{
::signal<void(EventData *)> events;
signals2};
struct Player
{
;
string nameint goals_scored = 0;
&game;
Game
(string name, Game &game) :
Player(std::move(name)), game(game)
name{}
void score()
{
++;
goals_scored{name, goals_scored};
PlayerScoreData ps.events(&ps);
game}
};
struct Coach
{
&game;
Game
explicit Coach(Game &game) : game(game)
{
.events.connect([](EventData *e) {
gameauto *ps = dynamic_cast<PlayerScoreData *>(e);
if (ps && ps->goals_scored_so_far < 3) {
<< "coach says: well done, " << ps->player_name << "\n";
cout }
});
}
};
int main()
{
;
Game game{"helywin", game};
Player player{game};
Coach coach.score();
player.score();
playerreturn 0;
}
中间者模式通过绑定触发事件和接收者来隔离两者之间的互相引用和依赖, 同时能够一对多多对一的进行通信
更高明的中间者实现使用事件允许参与者订阅和取消订阅系统中发生的事件. 组件和组件之间发送的消息可以视为events.
Memento
备忘录模式, 命令模式记录的变更允许回滚到系统任何时间点, 但有时候并不关心回放到系统的状态, 但是需要关心能回到之前的状态.
命令模式的存储的是修改过程, 备忘录模式存储的是修改后的状态
Bank Account
(int amount)
Memento deposit{
+= amount;
balance return { balance };
}
class Memento
{
int balance;
}
{ 100 };
BankAccount baauto m1 = ba.deposit(50);
auto m2 = ba.deposit(25);
<< ba << "\n"; // Balance: 175
cout
// undo to m1
.restore(m1);
ba<< ba << "\n"; // Balance: 150
cout
// redo
.restore(m2);
ba<< ba << "\n"; // Balance: 175 cout
Undo and Redo
单个账号状态备份, 用shared_ptr存储Memento
总结:
每个Memento对象都存储一个完整的数据能够把当前系统恢复到对应的状态
Null Object
纯抽象类用来做基类指针时, 比如Logger, 当并不需要类记录日志时, 没有适用的Logger类让其不记录日志
Null Object
这时需要一个NullLogger实现Logger中的函数并是个空函数
shared_ptr is not a Null Object
共享指针不是空对象
<int> n;
shared_ptrint x = *n + 1;
Design Improvements
提升BankAccount的使用性
- 在所有的地方检查指针
- 设置默认值
- 适用可选类型std::optional
Implicit Null Object
#include <memory>
#include <string>
#include <iostream>
using namespace std;
struct Logger
{
virtual ~Logger() = default;
virtual void info(const string &s) = 0;
virtual void warn(const string &s) = 0;
};
struct ConsoleLogger : Logger
{
void info(const string &s) override
{
<< "Info: " << s << endl;
cout }
void warn(const string &s) override
{
<< "Warn: " << s << endl;
cout }
};
struct NullLogger : Logger
{
void info(const string &s)
{};
void warn(const string &s)
{};
};
struct OptionalLogger : Logger
{
<Logger> impl;
shared_ptrstatic shared_ptr<Logger> no_logging;
explicit OptionalLogger(const shared_ptr<Logger> &logger) : impl{logger}
{}
virtual void info(const string &s) override
{
if (impl) impl->info(s);
}
virtual void warn(const string &s) override
{
if (impl) impl->warn(s);
}
};
<Logger> OptionalLogger::no_logging{};
shared_ptr
struct User
{
;
string name<OptionalLogger> logger;
shared_ptr
explicit User(string name, const shared_ptr<Logger> &logger = OptionalLogger::no_logging) :
(std::move(name)),
name{make_shared<OptionalLogger>(logger)}
logger{}
void log()
{
->info(name);
logger}
};
int main()
{
("u1");
User u1.log();
u1<ConsoleLogger> logger = make_shared<ConsoleLogger>();
shared_ptr("u2", logger);
User u2.log();
u2return 0;
}
std::optional的用法
用std::optional
为容器的变量可以具有bool操作符运算来判断值是否有效,
比如返回值是optional<string>
,
可以之接用if判断而不用调用值然后判断empty, 对于接口来说相对统一
Observer
C#等语言有开箱即用的, C++需要自己实现
观察者模式是一种行为型设计模式,它的作用是当一个对象的状态发生变化时,能够自动通知其他关联对象,自动刷新对象状态。观察者模式完美的将观察者和被观察的对象分离开
#include <iostream>
#include <vector>
using namespace std;
class Observer {
public:
virtual void update(int data) = 0;
};
class Subject {
public:
void attach(Observer* observer) {
.push_back(observer);
observers}
void detach(Observer* observer) {
for (auto it = observers.begin(); it != observers.end(); ++it) {
if (*it == observer) {
.erase(it);
observersreturn;
}
}
}
void notify(int data) {
for (auto observer : observers) {
->update(data);
observer}
}
private:
<Observer*> observers;
vector};
class ConcreteObserver : public Observer {
public:
(string name) : name(name) {}
ConcreteObservervoid update(int data) override {
<< "Observer " << name << " received data: " << data << endl;
cout }
private:
;
string name};
class ConcreteSubject : public Subject {
public:
void setData(int data) {
this->data = data;
(data);
notify}
private:
int data;
};
int main() {
;
ConcreteSubject subject("Observer 1");
ConcreteObserver observer1("Observer 2");
ConcreteObserver observer2.attach(&observer1);
subject.attach(&observer2);
subject.setData(10);
subject.detach(&observer1);
subject.setData(20);
subjectreturn 0;
}
Property Observers
C++17 Property
#include <cstdio>
class Person
{
int age_ = 0;
public:
int get_age() const { return age_; }
void set_age(int age) { age_ = age; }
__declspec(property(get=get_age, put=set_age)) int age;
void print() { printf("%d\n", age_); }
};
;
Person person.age = 10; person
其中set_age()
可以通知值的改变
Observer<T>
struct PersonListener
{
virtual void person_changed(Person &p, const string &property_name, const any new_value) = 0;
};
更好的解决办法, 不光针对Person
类型
template<typename T>
struct Observer
{
virtual void field_changed(T &source, const string &field_name) = 0;
};
struct ConsolePersonObserver : Observer<Person>
{
void field_changed(Person& source, const string &field_name) override
{
<< "Person's" << field_name << " has changed to " << source.get_age() << ".\n";
cout }
};
这样做的好处可以同时观察好几个类
struct ConsolePersonObserver : Observer<Person>, Observer<Creature>;
还可以用std::any
绑定任意类型,
比如类似void *
Observable<T>
当Person成为可被观察的类时, 需要实现以下职责:
- 用一个列表存储所有对Person类变化感兴趣的观察者们
- 允许观察者订阅和取消订阅(
subscribe()
/unsubscribe()
) - 当发生改变并发出
notify()
时通知所有观察者
原型如下:
template <typename T>
struct Observable
{
void notify(T &source, const string &name)
{
for (auto obs : observers)
{
->field_changed(source, name);
obs}
}
void subscribe(Observer<T>* f) { observers.push_back(f); }
void unsubscribe(Observer<T> *f) {...}
private:
<Observer<T> *> observers;
vector};
notify
函数在调用set_age
函数时使用,
但是要判断值是否真的改变了
Connecting Observers and Observables
所有代码的实现
#include <string>
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
struct Person;
template<typename T>
struct Observer
{
virtual void field_changed(T &source, const string &field_name) = 0;
};
template<typename T>
struct Observable
{
void notify(T &source, const string &name)
{
for (auto obs : observers) {
->field_changed(source, name);
obs}
}
void subscribe(Observer<T> *f)
{ observers.push_back(f); }
void unsubscribe(Observer<T> *f)
{
.erase(remove(observers.begin(), observers.end(), observer),
observers.end());
observers}
private:
<Observer<T> *> observers;
vector};
struct Person : Observable<Person>
{
int get_age() const
{ return age_; }
void set_age(int age)
{
if (age_ == age) return;
age_ = age;
(*this, "age");
notify}
() const
string get_name{ return name_; }
void set_name(const string &name)
{
if (name_ == name) return;
name_ = name;
(*this, "name");
notify}
private:
int age_;
name_;
string };
struct ConsolePersonObserver : Observer<Person>
{
void field_changed(Person &source, const string &field_name) override
{
// source.get_name写死有问题, 需要重载一个根据名称查找值的get_property(), 用std::variant
<< "Person's " << field_name
cout << " has changed to " << source.get_name() << ".\n";
}
};
int main()
{
;
Person p;
ConsolePersonObserver cpo.subscribe(&cpo);
p.set_name("Wang");
preturn 0;
}
Dependency Problems
当有一个值是依赖另一个值改变的时候, 代码就会很冗长,
比如can_vote
这个值实际不存在,
get的时候只是取age>18
,
设置age时这个值改变也要判断是否改变并发送通知, 这个问题很难得到解决
Unsubscription and Thread Safety
由于std::vector
是非线程安全的,
订阅和取消订阅不在一个线程中执行会出现问题, 解决办法是
- 在通知, 订阅, 取消订阅时加上锁
- 使用并发的vector, concurrent_vector
Reentrancy
函数可重入性, 当Person年龄大于17就自动取消订阅,
在field_changed
函数里面取消订阅, 调用链:
notify() --> field_changed() --> unsubscribe()
这在调用时由于加锁会导致无法取消订阅, notify和unsubscribe都加锁了, 解决办法有两种
- 避免出现这种情况
- 取消订阅不加锁, 只是把观察者设为空指针, 通知时也不加锁, 判断当前是否为空指针, 因为数组的增加和遍历是不会出现线程安全问题
void unsubscribe(Observer<T>* o)
{
auto it = find(observers.begin(), observers.end(), o);
if (it != observers.end())
*it = nullptr; // cannot do this for a set
}
//And, subsequently, when you notify(), you just need an extra check:
void notify(T& source, const string& name)
{
for (auto obs : observers)
if (obs)
->field_changed(source, name);
obs}
这样解决了notify和unsubscribe的锁竞争(contention), 但是同时订阅和取消订阅仍然会出问题
另外一种办法是通知的时候取一份列表的拷贝
void notify(T& source, const string& name)
{
<Observer<T>*> observers_copy;
vector{
<mutex_t> lock{ mtx };
lock_guard= observers;
observers_copy }
for (auto obs : observers_copy)
if (obs)
->field_changed(source, name);
obs}
还可以用recursive_mutex
替代mutex
,
但是很多开发者都讨厌使用, 会丢失性能, 而且通过改变代码结构能避免
还有一些问题:
- 如果观察者添加两次
- 如果能重复添加观察者, 那么取消订阅呢
- 如果改变容器, 使用std::set, std::unordered_set, 怎样实现
- 观察者有优先级
Observer via Boost.Signals2
使用Boost::Signal2的方案
template <typename T>
struct Observable
{
<void(T&, const string&)> property_changed;
signal};
struct Person : Observable<Person>
{
...
void set_age(const int age)
{
if (this->age == age) return;
this->age = age;
(*this, "age");
property_changed}
};
// 使用
{123};
Person pauto conn = p.property_changed.connect([](Person&, const string& prop_name)
{
<< prop_name << " has been changed" << endl;
cout });
.set_age(20); // name has been changed
p
// later, optionally
.disconnect(); conn
Summary
实现观察者模式的决定
- 决定要观察的信息
- 观察者的类是
- 是否每个观察者单独实现还是有一堆虚函数
- 是否需要处理取消订阅
- 如果不需要支持取消订阅,会节省很多时间, 因为不需要进行函数可重入性处理
- 如果要支持显式的取消订阅函数, 不能直接erase-remove掉, 而是标记元素待移除然后再去移除
- 如不希望直接使用指针,考虑使用weak_ptr
- 如果Observer<T>会被多个线程调用, 考虑保护下标数组
- 在每个相关的函数放上scoped_lock
- 使用线程安全的集合比如TBB/PPL concurrent_vector. 你会失去排序保证
- 如果允许多个下标代表同一来源, 那么不能使用std::set
很遗憾没有理想的观察者模式实现对应所有情况, 所以需要按照自己期望的来实现
State
状态模式, 状态是可以改变的, 重要的是谁触发了状态的改变, 有以下两种情况:
- 状态是带行为的类
- 状态和转移只是枚举. 我们用状态机来表述这些状态和过程
State-Driven State Transitions
简单的开关
要点:
状态并不是抽象的
其次状态允许从一个切换到另一个, 而且状态允许开关从一个状态到另一个状态而不是开关改变状态
最令人困惑的是, 默认的on/off状态表示我们已经在这个状态
state.hp
class LightSwitch;
struct State
{
virtual void on(LightSwitch *ls)
{ cout << "Light is already on\n"; }
virtual void off(LightSwitch *ls)
{ cout << "Light is already off\n"; }
};
struct OnState : State
{
()
OnState{ cout << "Light turned on\n"; }
void off(LightSwitch *ls) override;
};
struct OffState : State
{
()
OffState{ cout << "Light turned off\n"; }
void on(LightSwitch *ls) override;
};
class LightSwitch
{
*state;
State public:
()
LightSwitch{ state = new OffState(); }
void set_state(State *s)
{ this->state = s; }
void on()
{ state->on(this); }
void off()
{ state->off(this); }
};
state.cpp
#include <iostream>
using namespace std;
#include "state.hpp"
void OnState::off(LightSwitch *ls)
{
<< "Switching light off...\n";
cout ->set_state(new OffState);
lsdelete this;
}
void OffState::on(LightSwitch *ls)
{
<< "Switching light on...\n";
cout ->set_state(new OnState);
lsdelete this;
}
int main()
{
;
LightSwitch ls.on();
ls.off();
ls.off();
ls}
state diagram
stateDiagram-v2
on : Switch on
off : Switch off
on --> off: Turn off
off --> on: Turn on
on --> on: Already on
off --> off: Already off
Handmade State Machine
利用枚举和数据结构进行状态机的状态转移(不完全)
enum class State
{
,
off_hook,
connecting,
connected,
on_hold,
on_hook};
enum class Trigger
{
,
call_dialed,
hung_up,
call_connected,
placed_on_hold,
taken_off_hold,
left_message,
stop_using_phone};
// 包括了当前状态和可能出现的触发条件和触发条件后的结果
<State, vector<pair<Trigger, State>>> rules;
map
[State::off_hook] = {
rules{Trigger::call_dialed, State::connecting},
{Trigger::stop_using_phone, State::on_hook}
};
stateDiagram-v2
off_hook
connecting
connected
on_hold
on_hook
off_hook-->connecting: call_dialed
off_hook-->on_hook: stop_using_phone
connecting-->off_hook: hung_up
connecting-->connected: call_connected
connected-->on_hold: placed_on_hold
connected-->off_hook: hung_up
on_hold-->off_hook: hung_up
State Machines with Boost.MSM
Meta State Machine, 使用CRTP实现, boost库的MSM库和Statechart库
Summary
除了Boost.MSM和Boost.Statechart还有许多其他的状态机实现方法, 比如QStateMachine,QP等
除了状态机许多还支持Hierarchical State Machine(层级状态机)HSM, 例如MATLAB里面的StateFlow, 比如生病是一种状态, 生病这种状态包含了生不同的病的子状态
值得思考现代状态机和状态模式之间的差距, 不要为了实现形式上的接口设计很多笨重不直观的东西
Strategy
策略模式, 把实现一件事不同的步骤封装到不同的接口里面, 然后封装到通用不变的算法或方法里面, 在动态和静态典范里面存在着。
策略模式是一种行为型设计模式,它定义了一族算法,将每个算法都封装起来,并且使它们之间可以互换。这个模式使得算法的变化可以独立于使用它的客户端。C++提供了多态性来实现这个模式
Dynamic Strategy
比如一段文本要以markdown或者html方式输出, 类似的代码如下
// 抽象的处理方法
struct ListStrategy
{
virtual void start(ostringstream &oss) {}
virtual void add_list_item(ostringstream &oss, const string &item) {}
virtual void end(ostringstream &oss) {}
};
// 输出列表的通用方法抽象
struct TextProcessor
{
void append_list(const vector<string> &items)
{
->start(oss);
list_strategyfor (const auto &item : items) {
->add_list_item(oss, item);
list_strategy}
->end(oss);
list_strategy}
private:
;
ostringstream oss<ListStrategy> list_strategy;
unique_ptr};
struct HtmlListStrategy : ListStrategy {...};
struct MarkdownListStrategy : ListStrategy {...};
动态策略通过类的多态实现
Static Strategy
静态策略通过模板实现
template <typename LS>
struct TextProcessor
{
private:
;
ostringstream oss;
LS list_strategy}
// 使用
<MarkdownListStragegy> tpm;
TextProcessor.append_list(...); tpm
采用静态还是动态看需要
Template Method
模板方法模式, 加上策略模式和工厂模式很相似. 策略模式和模板方法模式不同的地方在于策略模式使用组合(不管动态还是静态), 而模板方法模式使用继承. 核心的原则是在一个地方定义算法的骨架在另外的地方实现剩余的细节, 符合开闭原则
Game Simulation
Chess有initialize
, startPlay
,
endPlay
等函数, ChineseChess继承自Chess,
run会调用这三个函数进行游戏,
而Chess需要做的只是继承前面三个函数而不用去管游戏怎么运行的,
相当于玩法流程骨架是Chess, 而实现算法的细节在实现具体的Chess类型
class Chess {
public:
void play() {
// 初始化游戏
();
initialize
// 开始游戏
();
startPlay
// 结束游戏
();
endPlay}
protected:
virtual void initialize() = 0;
virtual void startPlay() = 0;
virtual void endPlay() {};
};
class ChineseChess : public Chess {
protected:
void initialize() {
<< "初始化中国象棋游戏" << endl;
cout }
void startPlay() {
<< "开始中国象棋游戏" << endl;
cout }
};
class InternationalChess : public Chess {
protected:
void initialize() {
<< "初始化国际象棋游戏" << endl;
cout }
void startPlay() {
<< "开始国际象棋游戏" << endl;
cout }
void endPlay() {
<< "国际象棋游戏结束" << endl;
cout }
};
int main() {
*chineseChess = new ChineseChess();
Chess ->play();
chineseChess
*internationalChess = new InternationalChess();
Chess ->play();
internationalChess
return 0;
}
模板方法模式和策略模式都属于行为型设计模式,但它们有一些区别。以下是它们的区别:
- 在模板方法模式中,抽象类定义了一个算法的框架,但是一些步骤的实现由具体子类决定。这种模式的实现是在编译时发生的。而在策略模式中,不同的算法是在运行时选择的,通过将算法封装在独立的类中,使得它们可以相互替换。策略模式的实现是在运行时发生的。
- 模板方法模式中,抽象类是算法的核心,它定义了算法的骨架,具体实现由子类完成。而策略模式中,算法是核心,它被封装在独立的类中,并且可以在运行时选择。
- 模板方法模式通常只针对一套算法,注重对同一个算法的不同细节进行抽象提供不同的实现。相比之下,策略模式注重多套算法多套实现,算法之间不应该有交集,因此算法和实现是分开的。
Visitor
访问者模式, 当你获取了一个类型层次, 除非能够接触到源代码否则不可能在结构里面添加函数
表达式类
struct Expression {};
struct DoubleExpression : Expression
{
double value;
explicit DoubleExpression(const double value) : value{value} {};
};
struct AdditionExpression : Expression
{
*left, *right;
Expression (Expression *const left, Expression *const right) : left{left}, right{right}{};
AdditionExpression};
Instrusive Visitor
侵入式的访问者, 违反开闭原则, 直接修改代码, 要实现打印直接在Expression里面添加虚函数print
Reflective Printer
构造一个ExpressionPrinter, 传入Expression指针, 利用dynamic_cast判断当前指针类的具体子类类型, 使用此方法不需要在基类中增加接口也能实现特定功能的增加
#include <sstream>
#include <iostream>
using namespace std;
struct Expression
{
// dynamic_cast 需要析构函数为虚函数
virtual ~Expression() = default;
};
struct DoubleExpression : Expression
{
double value;
explicit DoubleExpression(const double value) : value{value} {};
};
struct AdditionExpression : Expression
{
*left, *right;
Expression (Expression *const left, Expression *const right) : left{left}, right{right} {};
AdditionExpression};
struct ExpressionPrinter
{
void print(Expression *e);
void print(DoubleExpression *de);
void print(AdditionExpression *ae);
() const { return oss.str(); }
string strprivate:
;
ostringstream oss};
// dynamic_cast的正确用法
void ExpressionPrinter::print(Expression *e)
{
if (auto de = dynamic_cast<DoubleExpression *>(e)) {
(de);
print} else if (auto ae = dynamic_cast<AdditionExpression *>(e)) {
(ae);
print}
}
void ExpressionPrinter::print(DoubleExpression *de)
{
<< de->value;
oss }
void ExpressionPrinter::print(AdditionExpression *ae)
{
<< "(";
oss (ae->left);
print<< "+";
oss (ae->right);
print<< ")";
oss }
int main()
{
auto ae = new AdditionExpression(new DoubleExpression(1.0), new DoubleExpression(2.0));
auto ae1 = new AdditionExpression(ae, new DoubleExpression(3));
;
ExpressionPrinter printer.print(ae1);
printer<< printer.str();
cout return 0;
}
WTH is Dispatch
dispatch(分派)的问题是找出需要调用哪一个函数, 特定的, 为了调用需要多少信息
举例子就是通过基类指针外部调用函数怎样映射到子类函数去
struct Pet {}
struct Cat : Pet {}
struct Dog : Pet {}
void func(Cat *cat) {}
void func(Dog *dog) {}
*pet = new Dog;
Pet (pet); func
这样造成的问题就是func不知道用哪个函数去实现, 所以只能通过dynamic_cast去判断, 另一种办法是用多态, 它能被正确的分派到必要的组件, 能够调用必要的重载. 也叫double dispatch(双分派), 因为:
- 在实际的对象上做了多态调用
- 在多态调用里面, 在调用重载. 在对象内, this指针是有一个精确的类型, 所以正确的重载也被调用
struct Pet {
virtual void call() = 0;
}
struct Cat : Pet {
void call() override { func(this); }
}
struct Dog : Pet {
void call() override { func(this); }
}
void func(Cat *cat) {}
void func(Dog *dog) {}
*pet = new Dog;
Pet ->call(); pet
在调用重载后的函数call之后会根据vtable自动定位到Dog::call, 从而this的类型也就清楚了
Classic Visitor
典型的访问者模式使用的是双分派, 约定的函数叫法:
- 访问者的成员函数叫visit()
- 贯穿层次成员函数实现叫accept()
表达式例子:
struct Expression
{
virtual void accept(ExpressionVisitor *visitor) = 0;
};
// 继承了Expression的类要实现accept
void accept(ExpressionVisitor *visitor)
{
->visit(this);
visitor}
struct ExpressionVisitor
{
virtual void visit(DoubleExpression *de) = 0;
virtual void visit(AdditionExpression *ae) = 0;
};
Implementing an Additional Visitor
好处就是只需要在层次(hierarchy)中实现一遍accept(), 你永远不需要再次接触层次的成员, 比如计算表达式。
class ElementA;
class ElementB;
class Element {
public:
virtual void accept(Visitor& visitor) = 0;
virtual void operation() = 0;
};
class ElementA : public Element {
public:
void accept(Visitor& visitor) override;
void operation() override { cout << "ElementA::operation()" << endl; }
};
class ElementB : public Element {
public:
void accept(Visitor& visitor) override;
void operation() override { cout << "ElementB::operation()" << endl; }
};
class Visitor {
public:
virtual void visit(ElementA& element) = 0;
virtual void visit(ElementB& element) = 0;
};
class ConcreteVisitor : public Visitor {
public:
void visit(ElementA& element) override { cout << "ConcreteVisitor::visit(ElementA&)" << endl; }
void visit(ElementB& element) override { cout << "ConcreteVisitor::visit(ElementB&)" << endl; }
};
int main() {
;
ElementA elementA;
ElementB elementB;
ConcreteVisitor visitor
.accept(visitor); // ConcreteVisitor::visit(ElementA&)
elementA.accept(visitor); // ConcreteVisitor::visit(ElementB&)
elementB
return 0;
}
Acyclic Visitor
Acyclic Visitor, 基于RTTI(Run-Time Type Information), 优点是不存在访问层次的限制, 但是牵涉性能问题
Acyclic Visitor模式是Cyclic Visitor模式的一种变体,它允许访问者访问对象的所有成员,但不包括继承来的成员。这种模式比较简单,只需要一次访问对象
Acyclic Visitor代码示例:
#include <iostream>
#include <sstream>
using namespace std;
template<typename Visitable>
struct Visitor
{
virtual void visit(Visitable &obj) = 0;
};
// 需要所有的数据模型都能接受visitor, 但是所有的特化都是唯一的, 需要引进marker interface, 带虚析构函数的空类
struct VisitorBase
{
virtual ~VisitorBase() = default;
};
struct Expression
{
virtual ~Expression() = default;
virtual void accept(VisitorBase &obj)
{
using EV = Visitor<Expression>;
// Expression类不执行
if (auto ev = dynamic_cast<EV *>(&obj))
->visit(*this);
ev}
};
struct DoubleExpression : Expression
{
double value;
explicit DoubleExpression(const double value) : value{value} {};
void accept(VisitorBase &obj) override
{
using EV = Visitor<DoubleExpression>;
if (auto ev = dynamic_cast<EV *>(&obj))
->visit(*this);
ev}
};
struct AdditionExpression : Expression
{
*left, *right;
Expression (Expression *const left, Expression *const right) : left{left}, right{right} {};
AdditionExpressionvoid accept(VisitorBase &obj) override
{
using EV = Visitor<AdditionExpression>;
if (auto ev = dynamic_cast<EV *>(&obj))
->visit(*this);
ev}
};
struct ExpressionPrinter : VisitorBase,
<DoubleExpression>,
Visitor<AdditionExpression>
Visitor{
void visit(DoubleExpression &obj) override;
void visit(AdditionExpression &obj) override;
() const { return oss.str(); }
string strprivate:
;
ostringstream oss};
void ExpressionPrinter::visit(DoubleExpression &obj)
{
<< obj.value;
oss }
void ExpressionPrinter::visit(AdditionExpression &obj)
{
<< "(";
oss .left->accept(*this);
obj<< "+";
oss .right->accept(*this);
obj<< ")";
oss }
int main()
{
auto ae = new AdditionExpression(new DoubleExpression(1.0), new DoubleExpression(2.0));
*ae1 = new AdditionExpression(ae, new DoubleExpression(3));
Expression auto printer = new ExpressionPrinter;
->accept(*printer);
ae1<< printer->str();
cout return 0;
}
Variants and std::visit
std::visit是标准库里面的模板, 配合std::variant可以存储不同的值, 类似于联合体
<string, int> house;
variant= 221;
house = "Castle"; house
完整例子
#include <variant>
#include <string>
#include <iostream>
#include <type_traits>
using namespace std;
struct AddressPrinter
{
void operator()(const string &house_name) const
{
<< "A house called " << house_name << endl;
cout }
void operator()(const int house_number) const
{
<< "House number " << house_number << endl;
cout }
};
int main()
{
using House = variant<string, int>;
{123};
House house;
AddressPrinter ap// 用visitor类
std::visit(ap, house);
// 用匿名表达式
std::visit([](auto &arg) {
// 当T是引用类型,decay<T>::type返回T引用的元素类型;当T是非引用类型,decay<T>::type返回T的类型
using T = decay_t<decltype(arg)>;
if constexpr (is_same_v<T, string>) {
<< "A house called " << arg.c_str() << endl;
cout } else {
<< "House number" << arg << endl;
cout }
}, house);
return 0;
}
可以参考[C++函数式编程]({% post_path ‘Cpp函数式编程’ %}#用模式匹配更好地处理代数数据类型)中使用overload模式利用模板和匿名函数构建访问者
可以配合``
Summary
访问者模式允许对任何层次的元素添加相同的方法, 实现包括
- Intrusive: 直接对基类添加虚函数, 每个类实现这个虚函数, 破坏了OCP原则
- Reflective: 增加visitor类, 用dynamic_cast实时派发
- Classic: (双派发), 整个层次需要更改, 但是只需要更改一次并且是通用的方法. 每个层次都需要学会accept() visitor. 然后子类化visitor保证所有层次的功能都能实现
访问者模式在串联的解释器模式中很常见: 解释了输入转化为对象, 渲染抽象的语法树就需要用ostringstream和所有的对象类型打交道
Appendix A: Functional Design Patterns
可以参考[C++函数式编程]({% post_path ‘Cpp函数式编程’ %})
单子(monad,也译单体)是函数式编程中的一种抽象数据类型,其特别之处在于,它是用来表示计算而不是数据的。在以函数式风格编写的进程中,单子可以用来组织包含有序操作的过程,或者用来定义任意的控制流(比如处理并发、异常、延续)
单子的构造包括定义两个操作bind和return,还有一个必须满足若干性质的类型构造器M
Maybe Monad
C++有很多方式表示不存在的值:
- 使用nullptr
- 使用智能指针, 可以测试是否缺少值
- std::optional<T>可以存储std::nullopt如果没有值
使用空指针需要每次使用时判断是否为空, Maybe的实现(编译不能通过)
#include <string>
#include <iostream>
using namespace std;
struct Address
{
*house_name = nullptr;
string };
struct Person
{
*address = nullptr;
Address };
template<typename T>
struct Maybe
{
*context;
T
(T *context) : context(context) {}
Maybe
template<typename Func>
auto With(Func evaluator)
{
!= nullptr ? maybe(evaluator(context)) : nullptr;
context }
template<typename TFunc>
auto Do(TFunc action)
{
if (context != nullptr) action(context);
return *this;
}
};
template<typename T>
auto maybe(T *context)
{
return Maybe<T>(context);
}
void print_house_name(Person *p)
{
auto maybe1 = maybe(p)
.With([](Person *x) { return x->address; })
.With([](Address *x) { return x->house_name; })
.Do([](auto x) { cout << *x << endl; });
}
int main()
{
auto a = new Address;
auto p = new Person;
->address = a;
p->house_name = new string("hello");
areturn 0;
}