QT 信号槽connect写法
QPushButton *btn = new QPushButton;
// 方式一:
connect(btn, &QPushButton::clicked, this, &MainWindow::close);
// 方式二:lambda表达式
connect(btn, &QPushButton::clicked, this, [&]() {
this->close();
});
QT4中老式写法,是利用SIGNAL SLOT宏,该方法已经过时了;
方式一 Qt5中新式写法,编译时对信号和槽进行检查不容易出错,而且随便一个普通函数都可作为槽,不必专门指定类型为槽函数;另外如果省略槽对象,qtcreator不会提示错误,但编译无法通过
该方法有个缺点:当函数有重载时会提示错误,解决方法可以是使用函数指针,例如:
void (QDoubleSpinBox::*f)(double) = &QDoubleSpinBox::valueChanged;
connect(ui->inflationRateDoubleSpinBox, f, this, &AverageWidget::refreshValue);
<QNonConstOverload>中有重载信号的辅助函数如下:(但是我的代码没有成功,)QOverload<double>(&QDoubleSpinBox::valueChanged)
方式二 采用了lambda表达式的写法,更加方便快捷。
参数要求:信号的参数数量 >= 槽的参数数量,且前面相同数量的参数类型应一致,信号中多余的参数会被忽略。
关于lambda需要注意一点:
QTimer::singleShot(3000, /* this, */ [&]{
this->close();
});
connect(btn, &QPushButton::clicked, /* this, */ [&]() {
this->close();
});
看下上面的示例,当我们用lambda表达式的时候,槽的接收者QObject是可以省略不写的,这时候Qt会默认发射者与接收者属于同一个QObject;
//connect to a functor
template <typename Func1, typename Func2>
static inline typename std::enable_if<QtPrivate::FunctionPointer<Func2>::ArgumentCount == -1, QMetaObject::Connection>::type
connect(const typename QtPrivate::FunctionPointer<Func1>::Object *sender, Func1 signal, Func2 slot)
{
return connect(sender, signal, sender, slot, Qt::DirectConnection);
}
当我们省略槽函数接收者QObject时,那我们就必须要注意lambda内成员的生命周期;例如示例的singleShot,若在槽函数响应前,this已经销毁变为无效指针,后果就会很严重!!!
为什么?
我们知道,connect的发射者与接收者任意一个销毁,那么这个connect就已经断开了;当我们省略接收者QObject的时候,发射者与接收者属于同一个QObject;在上面的示例中,信号槽connect关联依然存在,信号槽依然会触发,但此时this已经被销毁
lambda使用说明
代码如下:
int main()
{
int a = 1;
int b = 2;
auto func = \=, &b\->int {return b += a + c;};
return 0;
}
Lambda函数也就是一个函数,它的语法定义如下:
\capture\ mutable ->return-type{statement}
1.\[capture\]:捕捉列表。捕捉列表总是出现在Lambda函数的开始处。实际上,\[\]是Lambda引出符。编译器根据该引出符判断接下来的代码是否是Lambda函数。捕捉列表能够捕捉上下文中的变量以供Lambda函数使用;
2.(parameters):参数列表。与普通函数的参数列表一致。如果不需要参数传递,则可以连同括号“()”一起省略;
3.mutable:mutable修饰符。默认情况下,Lambda函数总是一个const函数,mutable可以取消其常量性。在使用该修饰符时,参数列表不可省略(即使参数为空);
4.->return-type:返回类型。用追踪返回类型形式声明函数的返回类型。我们可以在不需要返回值的时候也可以连同符号”->”一起省略。此外,在返回类型明确的情况下,也可以省略该部分,让编译器对返回类型进行推导;
5.{statement}:函数体。内容与普通函数一样,不过除了可以使用参数之外,还可以使用所有捕获的变量。
与普通函数最大的区别是,除了可以使用参数以外,Lambda函数还可以通过捕获列表访问一些上下文中的数据。具体地,捕捉列表描述了上下文中哪些数据可以被Lambda使用,以及使用方式(以值传递的方式或引用传递的方式)。语法上,在“\[\]”包括起来的是捕捉列表,捕捉列表由多个捕捉项组成,并以逗号分隔。捕捉列表有以下几种形式:
1.\[var\]表示值传递方式捕捉变量var;
2.\[=\]表示值传递方式捕捉所有父作用域的变量(包括this);
3.\[&var\]表示引用传递捕捉变量var;
4.\[&\]表示引用传递方式捕捉所有父作用域的变量(包括this);
5.\[this\]表示值传递方式捕捉当前的this指针。
上面提到了一个父作用域,也就是包含Lambda函数的语句块,说通俗点就是包含Lambda的“{}”代码块。上面的捕捉列表还可以进行组合,例如:
1.\[=,&a,&b\]表示以引用传递的方式捕捉变量a和b,以值传递方式捕捉其它所有变量;
2.\[&,a,this\]表示以值传递的方式捕捉变量a和this,引用传递方式捕捉其它所有变量。
不过值得注意的是,捕捉列表不允许变量重复传递。下面一些例子就是典型的重复,会导致编译时期的错误。例如:
3.\[=,a\]这里已经以值传递方式捕捉了所有变量,但是重复捕捉a了,会报错的;
4.\[&,&this\]这里&已经以引用传递方式捕捉了所有变量,再捕捉this也是一种重复。
下面是一段代码示例:
//遍历子窗口并显示
for(int i=0; i<windows.size(); ++i){
MdiChild *child = qobject_cast<MdiChild*>(windows.at(i)->widget());
QString text;
//窗口数小于9则设置编号为快捷键,这里&%1和%2之间不能有空格,否则无法实现快捷键效果
if(i<9){
text = tr("&%1%2").arg(1+i)
.arg(child->userFriendlyCurrentFile());
}else{
text = tr("%1%2").arg(1+i)
.arg(child->userFriendlyCurrentFile());
}
//添加动作到菜单,设置动作可选
QAction *action = ui->menuW->addAction(text);
action->setCheckable(true);
action->setChecked(child == activeMdiChild());
//下面的lambda不能以&的方式捕获,不然变量i的值只等于最终值
connect(action, &QAction::triggered,
[=](){this->setActiveSubWindow(windows.at(i));});
}
自定义信号:
信号必须是void类型,且只在.H中写定义,不在.CPP中写实现,若有参数,注意发射时要手动写参数。
使用方法:
可以手动发射;
也可以设置好触发条件,写个槽函数在槽中发射。(便于设置信号参数);
当然也可以直接作为槽函数执行发射动作(即,将想要发射的信号作为槽,适用无参数或参数相对简单固定)如:
//方法一,手动发射
//.H中
signals:
void signalDataEdited(int row, int column);
//.CPP中
emit signalDataEdited(AmountRow, currentColumn);
//方法二,自动发射,封装在槽中
//.H中
signals:
void signalDataEdited(int row, int column);
public:
void dataEdited();
//.CPP中 设置槽触发的条件,等于发射信号的条件
connect(ui->okBtn, &QPushButton::clicked,
this, &BalanceWidget::dataEdited);
//槽触发时功能只是发射信号
void BalanceWidget::dataEdited()
{
emit signalDataEdited(ui->tableWidget->currentRow(),
ui->tableWidget->currentColumn());
}
//方法三,自动发射,作为槽
connect(ui->okBtn, &QPushButton::clicked,
this, &BalanceWidget::signalDataEdited(ui->tableWidget->currentRow(),
ui->tableWidget->currentColumn()));
当枚举enum类型作为信号槽参数传递时,需要注册
//该类需要包含Q_OBJECT或者Q_GADGET才能注册
public:
enum SignalEnum {SINGLE_BALANCE, INTEGRATED_BALANCE};
Q_ENUM(SignalEnum)//需要在枚举声明的后边
其他:enum在元对象系统中存储格式为signed int
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。