防止在 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 应用程序中的信号阻塞,从而避免不必要的信号触发。

相关推荐
虾球xz3 分钟前
CppCon 2015 学习:3D Face Tracking and Reconstruction using Modern C++
开发语言·c++·学习·3d
林鸿群5 分钟前
C#子线程更新主线程UI及委托回调使用示例
开发语言·c#
奥修的灵魂18 分钟前
QT进阶之路:带命名空间的自定义控件在Qt设计器与qss中的使用技巧
qt·命名空间
Channing Lewis1 小时前
sql server如何创建表导入excel的数据
数据库·oracle·excel
秃头摸鱼侠1 小时前
MySQL安装与配置
数据库·mysql·adb
UGOTNOSHOT1 小时前
每日八股文6.3
数据库·sql
行云流水行云流水1 小时前
数据库、数据仓库、数据中台、数据湖相关概念
数据库·数据仓库
John Song1 小时前
Redis 集群批量删除key报错 CROSSSLOT Keys in request don‘t hash to the same slot
数据库·redis·哈希算法
IvanCodes2 小时前
七、Sqoop Job:简化与自动化数据迁移任务及免密执行
大数据·数据库·hadoop·sqoop