当前位置: 首页 > news >正文

【QA】观察者模式在QT有哪些应用?

1. 信号与槽机制

Qt的**信号与槽(Signals & Slots)**是观察者模式的典型实现,通过元对象系统(Meta-Object System)实现松耦合通信。

  • 核心特点

    • 类型安全:编译时检查参数匹配
    • 跨线程支持:通过Qt::QueuedConnection实现线程安全通信
    • 动态连接:运行时可添加/移除槽函数
  • 代码示例

    // 发射信号(带参数)
    emit dataFetched("Hello", timestamp); 
    
    // 连接信号与槽
    QObject::connect(button, &QPushButton::clicked, 
                     label, &QLabel::clear); 
    
  • Mermaid序列图

DataFetcher DataProcessor mDataLabel QWidget QApplication paintEvent dataFetched(data="Hello", timestamp=123) 发射信号 process_data(const QString&, int) setText("收到数据:%1 (%2秒)".arg(data).arg(timestamp)) 触发重绘请求 update() 显式触发界面刷新 postEvent(QPaintEvent) deliverEvent(QPaintEvent) 执行绘制逻辑 loop [[Qt事件循环]] DataFetcher DataProcessor mDataLabel QWidget QApplication paintEvent

2. 事件处理机制

Qt通过事件系统实现观察者模式,事件对象(QEvent)继承自QObject,通过重写事件处理函数响应特定事件。

  • 关键函数
    • mousePressEvent(QMouseEvent*):处理鼠标点击
    • keyPressEvent(QKeyEvent*):处理键盘输入
    • paintEvent(QPaintEvent*):处理界面重绘
  • 代码示例
    // 重写鼠标事件处理
    void MyWidget::mousePressEvent(QMouseEvent *event) {
        if (event->button() == Qt::LeftButton) {
            qDebug() << "左键点击坐标:" << event->pos(); // 
        }
    }
    
  • Mermaid类图
    处理
    1
    *
    QEvent
    +QEvent::Type type
    +QCoreApplication::postEvent()
    QMouseEvent
    +Qt::MouseButton button
    +QPoint pos
    QWidget
    +void mousePressEvent(QMouseEvent*)
    +void paintEvent(QPaintEvent*)

3. 事件过滤器(Event Filter)

通过installEventFilter实现跨控件事件监听,适用于非父子关系的对象通信。

  • 实现步骤
    1. 安装过滤器:widget->installEventFilter(this)
    2. 重写eventFilter(QObject*, QEvent*)
  • 代码示例
    // 安装事件过滤器
    label->installEventFilter(this); // 
    
    // 拦截Resize事件
    bool MyClass::eventFilter(QObject *obj, QEvent *event) {
        if (event->type() == QEvent::Resize) {
            qDebug() << "控件尺寸变化:" << obj->size(); // 
            return true; // 停止事件传播
        }
        return QWidget::eventFilter(obj, event);
    }
    
  • Mermaid序列图
    Widget MyClass installEventFilter() 事件触发时调用eventFilter() 处理事件(如Resize) 返回处理结果 Widget MyClass

4. 全局观察者模式

通过单例类实现跨层级消息传递,避免逐层转发。

  • 实现代码
    // 全局观察者类(单例)
    class GlobalObserver : public QObject {
    public:
        static GlobalObserver& instance() {
            static GlobalObserver observer;
            return observer;
        }
        
        void notify(const QString& msg) {
            QMetaObject::invokeMethod(receiver, "update", Qt::QueuedConnection,
                                    Q_ARG(QString, msg));
        }
        
    private:
        QList<QObject*> m_receivers;
    }; // 
    
  • 使用示例
    // 注册观察者
    GlobalObserver::instance().connect(this, &MyWidget::update);
    
    // 发送全局消息
    GlobalObserver::instance().notify("数据已更新"); // 
    
  • Mermaid类图
    观察者
    1
    *
    GlobalObserver
    +QList m_receivers
    +static GlobalObserver& instance()
    +void notify(const QString&)
    MyWidget
    +void update(const QString&)

5. 自定义观察者模式实现

手动实现接口和聚合关系,适用于复杂业务逻辑。

  • 代码示例
    // 抽象主题接口
    class ISubject {
    public:
        virtual void registerObserver(IObserver*) = 0;
        virtual void removeObserver(IObserver*) = 0;
        virtual void notifyObservers() = 0;
    };
    
    // 具体主题
    class WeatherData : public ISubject {
    private:
        QList<IObserver*> m_observers;
        double m_temperature;
    public:
        void registerObserver(IObserver* observer) override {
            m_observers.append(observer);
        }
        void notifyObservers() override {
            for (auto observer : m_observers) {
                observer->update(m_temperature);
            }
        }
    }; // 
    
  • Mermaid类图
    ISubject
    +registerObserver(IObserver*)
    +removeObserver(IObserver*)
    +notifyObservers()
    IObserver
    +void update(double)
    WeatherData
    +QList m_observers
    +double m_temperature
    +registerObserver(IObserver*)
    +notifyObservers()
    WeatherDisplay

总结

Qt通过信号与槽事件系统事件过滤器原生支持观察者模式,适用于UI交互、多线程通信等场景。对于复杂业务逻辑,可手动实现接口和聚合关系,或通过全局单例类实现跨层级通信。这些机制共同体现了Qt在解耦对象交互、提升扩展性方面的设计优势。

相关文章:

  • 【Linux篇】进程控制
  • Pytest的夹具
  • 夸克网盘突破限速下载
  • 【MySQL】内置函数
  • 11 python 数据容器-字符串
  • 面试题精选《剑指Offer》:JVM类加载机制与Spring设计哲学深度剖析-大厂必考
  • 九、JavaScript作用域、预解析
  • 【数据分享】2000—2024年我国乡镇的逐月归一化植被指数(NDVI)数据(Shp/Excel格式)
  • 遇到一个奇怪问题,页面请求不到后端
  • 基于SpringBoot+Vue3实现的宠物领养管理平台功能七
  • 95 克的工业级动能:STONE 80A-M 电调深度测评 —— 无人机动力系统的轻量化范式
  • 跨域问题确认及处理
  • windows10在wsl上利用GPU运行tensorflow 2.12
  • 基于PyCATIA的CATIA实体双侧分割技术实现与优化
  • Onlyoffice 编译打包运行过程优化
  • 基于CNN的FashionMNIST数据集识别4——GoogleNet模型
  • VMware主机换到高配电脑,高版本系统的问题
  • 视图窗口的客户区
  • Android Coil3 Fetcher preload批量Bitmap拼接扁平宽图,Kotlin
  • 【VolView】纯前端实现CT三维重建-CBCT
  • 媒体:演员黄杨钿甜耳环事件仍有三大疑问待解
  • 特朗普亲家有了新工作:美国驻法大使
  • 技术派|台军首次试射“海马斯”火箭炮,如何压制这种武器?
  • 中国戏剧梅花奖终评结果公示,蓝天和朱洁静等15名演员入选
  • 网络直播间销售玩具盲盒被指侵权,法院以侵犯著作权罪追责
  • 4年间职务侵占、受贿逾亿元,北京高院:严惩民企内部腐败