Qt 关于信号与槽函数实现机制浅谈

关于信号与槽函数

信号和槽函数是Qt用于实现对象之间互相通信、交互的一种机制。通过信号槽,同一线程内或多个线程间的对象可以把程序运行过程中的参数变量、状态参数等数据发送给其他对象,使其能够响应、运行相关函数以实现其功能。
信号和槽函数的绑定通过connect函数实现:

connect(sender, &Sender::signal, receiver, &Receiver::slot_function, type);

支持一对一和一对多绑定,绑定类型type分为:

  1. autoconnection:默认绑定方式,视绑定对象是否在同一线程内采用directconnection或queuedconnection;
  2. directconnection:槽函数会立即被执行,可能会阻塞UI线程;
  3. queuedconnection:槽函数在接受方线程中处理;
  4. blockedqueuedconnection:发送线程在槽函数执行完毕之前会阻塞,可能导致死锁;
  5. uniqueconnection:信号和槽函数只会生效一次。

由于信号和槽函数的生效对象间是解耦的,并且多线程间的发送和接收通过事件循环和事件队列实现,因此它可以适用于复杂场景,同时也是线程安全的。

关于信号与槽函数的实现机制浅谈

Qt通过元对象编译器MOC实现信号和槽函数。初学者在进行信号与槽函数编程时,会发现需要实现信号/槽的类都需要声明宏定义Q_OBJECT,这是因为在编译阶段,MOC会通过该宏定义把生成额外的元数据信息,用于绑定信号和槽函数。
举个例子,在窗口中创建两个QPushButton组件,并分别绑定两个槽函数:

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    init_connect();
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::init_connect()
{
    connect(ui->pushButton, &QPushButton::clicked, this, &MainWindow::slot_btn_clicked);
    connect(ui->pushButton_2, &QPushButton::clicked, this, &MainWindow::slot_btn2_clicked);
}

void MainWindow::slot_btn_clicked()
{
    qDebug() << "hello world";
}

void MainWindow::slot_btn2_clicked()
{
    qDebug() << "hello world2";
}
class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private:
    void init_connect();

private:
    Ui::MainWindow *ui;

private slots:
    void slot_btn_clicked();
    void slot_btn2_clicked();

由于在头文件中声明了宏定义Q_OBJECT,所以在编译过程中,编译器会生成moc_mainwindows.cpp文件,用来生成额外的元数据信息,下面是moc_mainwindows.cpp中的内容:

#include <memory>
#include "../../UIdemo/mainwindow.h"
#include <QtCore/qbytearray.h>
#include <QtCore/qmetatype.h>
#if !defined(Q_MOC_OUTPUT_REVISION)
#error "The header file 'mainwindow.h' doesn't include <QObject>."
#elif Q_MOC_OUTPUT_REVISION != 67
#error "This file was generated using the moc from 5.13.1. It"
#error "cannot be used with the include files from this version of Qt."
#error "(The moc has changed too much.)"
#endif

QT_BEGIN_MOC_NAMESPACE
QT_WARNING_PUSH
QT_WARNING_DISABLE_DEPRECATED
struct qt_meta_stringdata_MainWindow_t {
    QByteArrayData data[4];
    char stringdata0[47];
};
#define QT_MOC_LITERAL(idx, ofs, len) \
    Q_STATIC_BYTE_ARRAY_DATA_HEADER_INITIALIZER_WITH_OFFSET(len, \
    qptrdiff(offsetof(qt_meta_stringdata_MainWindow_t, stringdata0) + ofs \
        - idx * sizeof(QByteArrayData)) \
    )
static const qt_meta_stringdata_MainWindow_t qt_meta_stringdata_MainWindow = {
    {
QT_MOC_LITERAL(0, 0, 10), // "MainWindow"
QT_MOC_LITERAL(1, 11, 16), // "slot_btn_clicked"
QT_MOC_LITERAL(2, 28, 0), // ""
QT_MOC_LITERAL(3, 29, 17) // "slot_btn2_clicked"

    },
    "MainWindow\0slot_btn_clicked\0\0"
    "slot_btn2_clicked"
};
#undef QT_MOC_LITERAL

static const uint qt_meta_data_MainWindow[] = {

 // content:
       8,       // revision
       0,       // classname
       0,    0, // classinfo
       2,   14, // methods
       0,    0, // properties
       0,    0, // enums/sets
       0,    0, // constructors
       0,       // flags
       0,       // signalCount

 // slots: name, argc, parameters, tag, flags
       1,    0,   24,    2, 0x08 /* Private */,
       3,    0,   25,    2, 0x08 /* Private */,

 // slots: parameters
    QMetaType::Void,
    QMetaType::Void,

       0        // eod
};

void MainWindow::qt_static_metacall(QObject *_o, QMetaObject::Call _c, int _id, void **_a)
{
    if (_c == QMetaObject::InvokeMetaMethod) {
        auto *_t = static_cast<MainWindow *>(_o);
        Q_UNUSED(_t)
        switch (_id) {
        case 0: _t->slot_btn_clicked(); break;
        case 1: _t->slot_btn2_clicked(); break;
        default: ;
        }
    }
    Q_UNUSED(_a);
}

QT_INIT_METAOBJECT const QMetaObject MainWindow::staticMetaObject = { {
    &QMainWindow::staticMetaObject,
    qt_meta_stringdata_MainWindow.data,
    qt_meta_data_MainWindow,
    qt_static_metacall,
    nullptr,
    nullptr
} };


const QMetaObject *MainWindow::metaObject() const
{
    return QObject::d_ptr->metaObject ? QObject::d_ptr->dynamicMetaObject() : &staticMetaObject;
}

void *MainWindow::qt_metacast(const char *_clname)
{
    if (!_clname) return nullptr;
    if (!strcmp(_clname, qt_meta_stringdata_MainWindow.stringdata0))
        return static_cast<void*>(this);
    return QMainWindow::qt_metacast(_clname);
}

int MainWindow::qt_metacall(QMetaObject::Call _c, int _id, void **_a)
{
    _id = QMainWindow::qt_metacall(_c, _id, _a);
    if (_id < 0)
        return _id;
    if (_c == QMetaObject::InvokeMetaMethod) {
        if (_id < 2)
            qt_static_metacall(this, _c, _id, _a);
        _id -= 2;
    } else if (_c == QMetaObject::RegisterMethodArgumentMetaType) {
        if (_id < 2)
            *reinterpret_cast<int*>(_a[0]) = -1;
        _id -= 2;
    }
    return _id;
}
QT_WARNING_POP
QT_END_MOC_NAMESPACE

其中qt_metacall和qt_static_metacall函数通过为槽函数绑定了其在调用过程中的相关参数(发送对象、id等),从而实现了对指定槽函数的调用。

  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值