lambda qt 参数 槽函数_Qt信号槽

本文详细介绍了QT5中信号槽的两种新式写法,包括使用函数指针和Lambda表达式。Lambda表达式使得信号槽连接更加灵活,但需要注意槽函数接收者生命周期和参数匹配问题。同时,文章提供了Lambda函数的语法解析和使用注意事项,并给出了自定义信号的创建及发射方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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);

中有重载信号的辅助函数如下:(但是我的代码没有成功,)

QOverload(&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

static inline typename std::enable_if<:functionpointer>::ArgumentCount == -1, QMetaObject::Connection>::type

connect(const typename QtPrivate::FunctionPointer::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

MdiChild *child = qobject_cast(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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值