https://www.devbean.net/2012/08/qt-study-road-2-signal-slot/
QMetaObject::Connection connect(const QObject *, const char *,
const QObject *, const char *,
Qt::ConnectionType);
QMetaObject::Connection connect(const QObject *, const QMetaMethod &,
const QObject *, const QMetaMethod &,
Qt::ConnectionType);
QMetaObject::Connection connect(const QObject *, const char *,
const char *,
Qt::ConnectionType) const;
QMetaObject::Connection connect(const QObject *, PointerToMemberFunction,
const QObject *, PointerToMemberFunction,
Qt::ConnectionType)
QMetaObject::Connection connect(const QObject *, PointerToMemberFunction,
Functor);
第一个,sender 类型是const QObject *
,signal 的类型是const char *
,receiver 类型是const QObject *
,slot 类型是const char *
。这个函数将 signal 和 slot 作为字符串处理。
第二个,sender 和 receiver 同样是const QObject *
,但是 signal 和 slot 都是const QMetaMethod &
。我们可以将每个函数看做是QMetaMethod
的子类。因此,这种写法可以使用QMetaMethod
进行类型比对。
第三个,sender 同样是const QObject *
,signal 和 slot 同样是const char *
,但是却缺少了 receiver。这个函数其实是将 this 指针作为 receiver。
第四个,sender 和 receiver 也都存在,都是const QObject *
,但是 signal 和 slot 类型则是PointerToMemberFunction
。看这个名字就应该知道,这是指向成员函数的指针。
第五个,前面两个参数没有什么不同,最后一个参数是Functor
类型。这个类型可以接受 static 函数、全局函数以及 Lambda 表达式。
SIGNAL
和SLOT
这两个宏,将两个函数名转换成了字符串
槽的形参数目可以比信号少。
常见写法:
1、无参信号->无参槽:
connect(tcpSkt, SIGNAL(readyRead()), this, SLOT(when_tcpReceivedData()));
2、有参信号->有参槽:
connect(tcpSkt, SIGNAL(error(QAbstractSocket::SocketError)), this, SLOT(when_tcpError(QAbstractSocket::SocketError )));
3、有参信号->部分参数槽
也即:槽可以接收信号的部分参数,但是只能依次忽略后面的参数。
connect(pA, SIGNAL(info(int,char,bool)), this, SLOT(when_rcvInfo(int,char)));
4、以上3种改用函数指针
connect(tcpSkt, &QTcpSocket::readyRead, this, &MainWindow::when_tcpReceivedData));
该代码的作用与前面的1,完全一致。
5、上述4的困境
如果信号存在重载、槽存在重载,那么4的代码将会编译报错,因为编译器不知道你要连接哪个信号。解决方案是,用函数器封装一下:
connect(tcpSkt, QOverload<QAbstractSocket::SocketError>::of(&QTcpSocket::error), this, &MainWindow::when_tcpError);
上述代码将根据QOverload后的参数表,来识别QTcpSocket::error的重载