在度娘参阅了各大博客,都没有找的这个问题的解决方案,要么就是解决办法是错误的。最后还是在stackoverflow中找到的。难道就我一个人遇到过这个问题吗。。。
问题描述
国内博客普遍认为,像以下这种信号槽连接,是可以直接调用disconnect()
函数来取消连接的:
QTimer* tm = new QTimer(this);
connect(tm, &QTimer::timeout, this, [=]()
{
qDebug()<<"fuck";
tm->disconnect();
});
然而,经过测试,这种办法是完全错误的,调用disconnect()
后,触发timeout后还是会输出"fuck",如果多次调用这串代码,会创建更多的连接,而且这些连接是无法被捕获的,类似于申请了一块内存空间,而没有用指针指向它。
经过分析,我认为disconnect()
函数只能销毁代码中定义的命名函数,而对于匿名函数是无能为力的,需要我们自己去解除连接。所以,我们需要注意到connect()
的返回值,利用该返回值来定位对应的连接。
解决办法
QTimer* tm = new QTimer(this);
auto conn = std::make_shared<QMetaObject::Connection>();
*conn = connect(tm, &QTimer::timeout, this, [=]()
{
qDebug()<<"fuck";
disconnect(*conn);
});
参考:
https://stackoverflow.com/questions/14828678/disconnecting-lambda-functions-in-qt5