防止在 Qt 中触发信号

在 Qt 中工作时,有时我们需要暂时阻止某些信号的触发。以下是一个经典场景:我们有一个 QCheckBox 对象,当用户勾选或取消勾选时,需要调用一个函数,因此我们将这个函数连接到 stateChanged(int state) 信号。然而,在某些条件下,我们在代码中更改 QCheckBox 的状态,这会导致触发不需要的信号。那么如何在特定情况下防止信号触发呢?

使用 clicked 信号

如果你只想在用户实际点击复选框时触发信号,可以使用 clicked 信号,因为这个信号只在用户点击复选框时触发,而不会在代码中使用 setChecked 方法更改状态时触发。

使用 QObject::blockSignals

你可以使用 QObject::blockSignals 方法来暂时阻止信号的发射:

cpp 复制代码
bool oldState = checkBox->blockSignals(true);
checkBox->setChecked(true);
checkBox->blockSignals(oldState);

这种方法的缺点是它会阻止所有信号的发射,但对于 QCheckBox 来说,这通常不是什么大问题。

使用 RAII 封装 blockSignals

为了更好地管理信号阻塞,可以使用 RAII(资源获取即初始化)模式。以下是一个示例类:

cpp 复制代码
class SignalBlocker
{
public:
    SignalBlocker(QObject *obj) : m_obj(obj), m_old(obj->blockSignals(true)) {}
    ~SignalBlocker() { m_obj->blockSignals(m_old); }

private:
    QObject *m_obj;
    bool m_old;
};

使用这个类,阻塞和恢复信号变得更加自动化和安全,特别是在异常处理的情况下。

使用 Qt5.3 引入的 QSignalBlocker

从 Qt5.3 开始,引入了 QSignalBlocker 类,可以简化信号的阻塞:

cpp 复制代码
if (something) {
   const QSignalBlocker blocker(someQObject);
   // 在这里不会发射信号
}

这个类不仅方便而且异常安全。

对多个对象进行信号阻塞

如果需要同时对多个对象进行信号阻塞,可以扩展 SignalBlocker 类:

cpp 复制代码
class SignalBlocker
{
public:
    SignalBlocker(QObject *obj) {
        insert(QList<QObject*>() << obj);
    }
    SignalBlocker(QList<QObject*> objects) {
        insert(objects);
    }
    void insert(QList<QObject*> objects) {
        for (auto obj : objects)
            m_objs.insert(obj, obj->signalsBlocked());
        blockAll();
    }
    void blockAll() {
        for(auto m_obj : m_objs.keys())
            m_obj->blockSignals(true);
    }
    ~SignalBlocker() {
        for(auto m_obj : m_objs.keys())
            m_obj->blockSignals(m_objs[m_obj]);
    }

private:
    QMap<QObject*,bool> m_objs;
};

使用示例:

cpp 复制代码
void SomeType::myFunction() {
    SignalBlocker blocker(QList<QObject*>()
                          << m_paramWidget->radioButton_View0
                          << m_paramWidget->radioButton_View1
                          << m_paramWidget->radioButton_View2);
    // 执行其他操作
}

通过这些方法,你可以灵活地管理 Qt 应用程序中的信号阻塞,从而避免不必要的信号触发。

相关推荐
Pocker_Spades_A7 分钟前
Python快速入门专业版(二十九):函数返回值:多返回值、None与函数嵌套调用
服务器·开发语言·python
良木林10 分钟前
浅谈原型。
开发语言·javascript·原型模式
烈风21 分钟前
004 Rust控制台打印输出
开发语言·后端·rust
恣艺41 分钟前
Redis列表(List):实现队列/栈的利器,底层原理与实战
数据库·redis·list
一枝小雨1 小时前
【C++】list 容器操作
开发语言·c++·笔记·list·学习笔记
HMBBLOVEPDX1 小时前
C++(继承和多态)
开发语言·c++·继承和多态
yvya_1 小时前
JVM介绍
java·开发语言·jvm
秋难降1 小时前
零基础学习SQL(十一):SQL 索引结构|从 B+Tree 到 Hash,面试常问的 “为啥选 B+Tree” 有答案了
数据库·后端·mysql
代码的余温1 小时前
Linux内核调优实战指南
linux·服务器·数据库
almighty271 小时前
C# DataGridView表头自定义设置全攻略
数据库·c#·winform·datagridview·自定义表头