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

123654_
81 声望5 粉丝

君子曰:学不可以已。