网站首页 > 技术文章 正文
信号槽:
信号槽机制最初是有QT框架引入,是一种广泛应用于事件通知的机制。信号和槽是一对通信机制,信号是一个事件的通知,槽是一个响应信号的回调函数。
观察者模式:
观察者模式是一种行为型设计模式,用于建立一种一对多的依赖关系,让多个观察者对象能够自动接收到主题对象(被观察者)的状态变化通知,并且通常能自发地更新自己的状态。
传统的观察者模式实现案例如下:可以发现是被观察者持有了多个观察者对象,当状态信息改变的时候,逐个通知观察对象,显然这样实现被观察者和观察者是聚合关系,还是比较耦合的,信号槽的方式来实现则比较松散。
#include <iostream>
#include <list>
#include <algorithm>
// 观察者基类
class Observer {
public:
virtual ~Observer() {}
virtual void update(int value) = 0;
};
// 具体观察者
class ConcreteObserver : public Observer {
private:
std::string name;
public:
ConcreteObserver(const std::string& n) : name(n) {}
void update(int value) override {
std::cout << "Observer " << name << " reacts to the event with value: " << value << std::endl;
}
};
// 主题基类
class Subject {
private:
std::list<Observer*> observers;
protected:
void attach(Observer* o) {
observers.push_back(o);
}
void detach(Observer* o) {
observers.remove(o);
}
void notify() {
for (Observer* o : observers) {
o->update(10); // 假设更新的值为10
}
}
};
// 具体主题
class ConcreteSubject : public Subject {
private:
int state;
public:
void setState(int s) {
state = s;
notify();
}
int getState() {
return state;
}
};
int main() {
ConcreteSubject subject;
ConcreteObserver observer1("A");
ConcreteObserver observer2("B");
subject.attach(&observer1);
subject.attach(&observer2);
subject.setState(10); // 改变状态,通知观察者
subject.detach(&observer1);
subject.setState(20); // 再次改变状态,通知剩余的观察者
return 0;
}
案例:
以下是一个基于信号槽实现的观察者模式案例,被观察者并不需要持有观察者对象。
这个案例是很简单的一个实现,目的也是为了理解信号槽这种歌解耦的设计思想。
在实际项目中,通常我们会做两点的升级:
--1 使用模板类和模板函数来实现信号和槽函数,提升代码的通用性
--2 信号触发有两种方式,一种是异步的,一种是同步的,我们这里明显是同步的,但实际项目中有可能任务消费是在其他的线程中执行的
#include <functional>
#include <iostream>
#include <vector>
class Slot {
public:
explicit Slot(std::function<void()> f) : f_(f) {}
void Call() { f_(); }
private:
std::function<void()> f_;
};
class Signal {
public:
void connect(Slot &slot) { slots_.emplace_back(slot); }
void Emit() {
for (auto &slot : slots_) {
slot.Call();
}
}
private:
std::vector<Slot> slots_;
};
class Observer {
public:
explicit Observer(const std::string &name) : name_(name) {
std::cout << "Observer " << this << " created" << std::endl;
}
void Notify() {
std::cout << "Observer " << this << " notified" << std::endl;
}
private:
std::string name_;
};
class Subject {
public:
void addObserver(Observer *observer) {
Slot slot(std::bind(&Observer::Notify, observer));
signal_.connect(slot);
}
void notifyObservers() { signal_.Emit(); }
private:
Signal signal_;
};
int main(int argc, char const *argv[]) {
Subject subject;
Observer observer1("Observer 1");
Observer observer2("Observer 2");
subject.addObserver(&observer1);
subject.addObserver(&observer2);
std::cout << "Notifying observers..." << std::endl;
subject.notifyObservers();
return 0;
}
猜你喜欢
- 2025-01-08 嵌入式中,日志调试法的一些规则!
- 2025-01-08 一行代码改进:Logtail的多行日志采集性能提升7倍的奥秘
- 2025-01-08 嵌入式大杂烩周记 第 7 期:zlog
- 2025-01-08 C语言总结:C语言字符串练习题(十二种习题示例)
- 2025-01-08 C语言100题集合027-二维数组的经典案例,非常重要
- 2025-01-08 一个例子让你看清线程调度的随机性
- 2025-01-08 C++17:结构化绑定
- 2025-01-08 C语言中main函数详解
- 2025-01-08 64TB硬盘容量测试程序(C++)
- 2025-01-08 使用CyberRT写第一个代码
- 最近发表
-
- 用Cursor开启JAVA+AI生涯_javascirpt怎么开启
- 大数据调度服务监控平台_大数据调度是什么意思
- SpringBoot、MyBatis、Vue搭建一个Java企业应用开源框架源码分享
- 大数据技术之Flume_大数据volume的含义
- Jenkins运维之路(Slave容器节点)_jenkins slave工作原理
- 程序员自救指南:IDEA 卡成狗?我的 9G 堆内存调参表让你起飞 附避坑
- JMeter:一个简单的测试计划怎么做?
- Windows 命令行终端 PowerShell 美化计划
- JDK25即将发布!新特性概览_jdk52.0
- JDK 25 新特性极简总结(2025 年 9 月 16 日发布,LTS 长期支持)
- 标签列表
-
- cmd/c (90)
- c++中::是什么意思 (84)
- 标签用于 (71)
- 主键只能有一个吗 (77)
- c#console.writeline不显示 (95)
- pythoncase语句 (88)
- es6includes (74)
- sqlset (76)
- apt-getinstall-y (100)
- node_modules怎么生成 (87)
- chromepost (71)
- flexdirection (73)
- c++int转char (80)
- mysqlany_value (79)
- static函数和普通函数 (84)
- el-date-picker开始日期早于结束日期 (76)
- js判断是否是json字符串 (75)
- c语言min函数头文件 (77)
- asynccallback (87)
- localstorage.removeitem (77)
- vector线程安全吗 (70)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- 无效的列索引 (74)