lambda写法
Qt 中信号槽lambda表达式优缺点: 好处
坏处
C++ 中lambda表达式构成
[capture](parameters) mutable ->return-type
{
statement
}
[函数捕获列表](操作符重载函数参数)mutable ->返回值{函数体}
- [capture]:函数捕获列表。始终出现在Lambda函数的开始处。[]是Lambda引出符。编译器根据该引出符来分析后续代码是否是Lambda函数。函数捕获列表能够捕捉上下文中的变量以供匿名函数使用;
- (parameters):操作符重载函数参数。和普通函数的参数列表一致。如果不需要参数传递,则可以连同括号“()”一起省略;
- mutable:mutable修饰符。默认情况下,Lambda函数是一个const函数,mutable可以取消其常量性。在使用该修饰符时,参数列表不可省略(即使参数为空);
- ->return-type:返回类型。我们可以在不需要返回值,则可以连同符号”->”一起省略。并且,在返回类型明确的情况下,也可以省略该部分,编译器会根据返回类型进行推导;
- {statement}:函数体。内容与普通函数一致,除了可以使用参数之外,还可以使用捕获列表中的变量
捕获列表[]的几种情况
- [this]捕捉当前对象的this指针,让 Lambda 表达式拥有和当前类成员同样的访问权限;
- [=]表示值传递方式捕捉所有父作用域的变量(包括this);
- [&]表示引用传递方式捕捉所有父作用域的变量(包括this);
- [var]表示值传递方式捕捉变量var,函数体内不能修改传递进来的var的拷贝,如果想修改必须加上mutable;
- [&var]表示引用传递捕捉变量var;
- [vara,&varb]值传递方式捕捉变量vara,引用传递捕捉变量varb;
- [this,&varb]捕捉当前对象的this指针,varb则按引用传递捕捉变量;
- [=,&vara,&varb] vara和varb按引用进行传递外,其他参数按值进行传递
lambda 常见错误 当省略调接收者时,lambda中无法捕获发送者是谁
connect(ui.pushButton, &QPushButton::clicked, [this]() {
QPushButton* btn_ptr = qobject_cast<QPushButton*>(QObject::sender());
qDebug() << __FUNCTION__ << btn_ptr;
});
使用QTimer::singleShot,可能会导致崩溃
void TestMoc::testMocSlot()
{
QTimer::singleShot(5000, this, [this]()
{
});
QTimer::singleShot(5000, [this]()
{
});
}
以上区别在于第二个参数,为了访问类成员,lambda都捕获this指针。 方式1:第二个参数接受对象传入this指针,若Qobject对象在超时前销毁,超时后也无法调用lambda,所以不会奔溃。 方式2:如果QObject对象在超时前已经销毁,则超时时仍会调用lambda,而lambda捕获了this指针,当访问成员变量或函数时,这时导致奔溃。
队列连接 引用传递问题
当在主线程和子线程中,通过信号与槽发送QString,遇到以下问题
QObject::connect: Cannot queue arguments of type 'QString&'
原因:
解决方法: 改成const引用或值传递
signals:
void testMocSignal2(QString &str,int input);
protected slots:
void testMocSlot2(QString& str, int input);
connect(this,&TestMoc::testMocSignal2, this, &TestMoc::testMocSlot2,Qt::QueuedConnection);
signals:
void testMocSignal2(const QString &str,int input);
protected slots:
void testMocSlot2(const QString& str, int input);
connect(this,&TestMoc::testMocSignal2, this, &TestMoc::testMocSlot2,Qt::QueuedConnection);
····
|