Qt 信号槽

写在前面

    在界面搭建时,我们需要很多触发事件。比如,鼠标右击,我们希望弹出对话框;点击退出的按钮,界面就会关闭或者执行其他的操作(如下图)。在Qt当中,提供了信号槽的运行机制。


右键弹出菜单栏
按钮动作

1 元对象系统

    信号和槽机制是Qt 的核心特征,需要借助Qt 的元对象系统实现。除了支持信号槽机制,元对象系统提供的功能有:对象间通信的运行时类型信息和动态属性系统等。

    Qt的元对象系统基于如下三个关键要素:
(1)类——QObject:所有元对象系统的基类

(2)宏——Q_OBJECT:为 Q_OBJECT 宏展开后所声明的成员函数的成生实现代码识别 Qt中特殊的关键字及宏,比如识别出 Q_PROPERTY宏、Q_INVOKABLE宏、slot、signals等

    放在继承自QObject的某个类私有区域中,使其拥有元对象特性:信号槽、动态特性等。

(3)元对象编译器——moc(Meta-Object Compiler):Qt的元对象编译器(MOC)是一个预处理器,在源程序被编译前先将Qt特性的程序转换为标准C++兼容的形式,然后再由标准C++编译器进行编译。(为每个QObject 的子类,提供实现了元对象特性所必须的代码。)

2 信号-槽

(1)槽的参数的类型需要与信号参数的类型相对应,
(2)槽的参数不能多余信号的参数,因为若槽的参数更多,则多余的参数不能接收到信号传递过来的值,若在槽中使用了这些多余的无值的参数,就会产生错误。
(3)若信号的参数多余槽的参数,则多余的参数将被忽略。
(4)一个信号可以与多个槽关联,多个信号也可以与同一个槽关联,信号也可以关联到另一个信号上。
(5)若一个信号关联到多个槽时,则发射信号时,槽函数按照关联的顺序依次执行。
(6)若信号连接到另一个信号,则当第一个信号发射时,会立即发射第二个信号。

2-1简介

1 信号

定义的形式如下:
signals:
void mySignal();
void mySignal(int x);
void mySignalParam(int x,int y);

(1)信号使用 signals 关键字声明,在其后面有一个冒号“:”,在其前面不能有 public、private、 protected 访问控制符,信号默认是 public 的。
(2)信号只需像函数那样声明即可,其中可以有参数,参数的主要作用是用于和槽的通信,这就像普通函数的参数传递规则一样。信号虽然像函数,但是对他的调用方式不一样,信号需要使用 emit 关键字发射。
(3)信号只需声明,不能对其进行定义,信号是由 moc 自动生成的。

2 槽函数

形如:
public slots:
void mySlot();
void mySlot(int x);
void mySignalParam(int x,int y);

(1)声明槽需要使用 slots 关键字,在其后面有一个冒号“:”,且槽需使用 public、private、 protected 访问控制符之一。
(2)槽就是一个普通的函数,可以像使用普通函数一样进行使用,槽与普通函数的主要区别是,槽可以与信号关联。

3 发射信号

(1)发射信号需要使用emit关键字,注意,在 emit 后面不需要冒号。
(2)emit 发射的信号使用的语法与调用普通函数相同,比如有一个信号为 void f(int),则发送的语法为:emit f(3);
(3)当信号被发射时,与其相关联的槽函数会被调用( 注意:信号和槽需要使用
QObject::connect 函数进行关联之后,发射信号后才会调用相关联的槽函数)。
(4)注意:因为信号位于类之中,因此发射信号的位置需要位于该类的成员函数中或该类能见到信号的标识符的位置。

4 自定义信号槽实例

示例:创建信号和槽 
//头文件 m.h 的内容#ifndef    M_H #define M_H #include<QObject> #include <iostream> using namespace std; 
 
class A:public QObject
{ 
   //信号和槽必须继承自 QObject 类
   Q_OBJECT // 必 须 添 加 该 宏 

  //public signals:void s1(int); //错误 signals 前不能有访问控制符。 
     signals:void s(); //使用 signals 关键字声明信号,信号的语法与声明函数相同。 
     signals:void s(int,int); //正确,信号可以有参数,也可以重载。 
        //void s2(){} //错误,信号只需声明,不能定义。 
       void s3(); //注意:这仍是声明的一个信号 
public: //信号声明结束后,重新使用访问控制符,表示以下声明的是成员函数。 
        void g()
      { 
    emit s3();  /*发射信号,其语法与调用普通函数相同,在信号与槽关联之前,发射的信号不会调用相应的槽函数。*/ 
    // emit: s3(); //错误,emit 后不能有冒号。 
       }
}; 
 
class B:public QObject
{ 
Q_OBJECT 
public slots: // 使 用 slots 关 键 字 声 明 槽 
void x(){cout<<"X"<<endl;} /*正确,槽就是一个普通函数,只是需要使用 slots 关键字,且
能和信号相关联。*/ 
        //slots: void x(){} //错误,声明槽时需要指定访问控制符。 
     public: 
         void g()
         { // emit s3(); //错误,在类 B 中对于标识符 s3 是不可见的 
          } 
}; #endif // M_H 

//源文件的内容
#include "m.h" 
int main(int argc, char *argv[])
 { 
    A ma; B mb; 
    QObject::connect(&ma,&A::s3,&mb,&B::x); //关联信号和槽,详见后文 
    ma.g(); //调用对象 mb 的成员函数 x 输出 X,可见对象 ma 和 mb 之间实现了通信。 
    return 0; 
 } 

5 补充说明

  1. 信号和槽是用于对象之间的通信的,这是 Qt 的核心。为此 Qt 引入了一些关键字,他们是slots、signals、emit,这些都不是 C++关键字,是 Qt 特有的,这些关键字会被 Qt 的moc 转换为标准的 C++语句。

  2. 只有QObject 及其派生类才能使用信号和槽机制,且在类之中还需要使用Q_OBJECT 宏。

  3. Qt 的部件类中有一些已经定义好了的信号和槽,也可以子类化部件类,然后添加自已的信号和槽。

  4. 信号与槽本质上式回调函数。信号与槽的效率是非常高的,但是同真正的回调函数比较起来,由于增加了灵活性,因此在速度上还是有所损失。

  5. 信号与槽机制与普通函数的调用一样,如果使用不当的话,在程序执行时也有可能产生死循环。因此,在定义槽函数时一定要注意避免间接形成无限循环,即在槽中再次发射所接收到的同样信号。例如在 mySlot() 槽函数中加上语句 emit mySignal() 即可形成死循环。

  6. 如果一个信号与多个槽相联系的话,那么,当这个信号被发射时,与之相关的槽被激活的顺序将是随机的。

2-2连接信号和槽——connect

1 表达形式

static QMetaObject::Connection connect(const QObject *sender, const char *signal,
const QObject *receiver, const char *method,Qt::ConnectionType type = Qt::AutoConnection)
各参数意义如下:
(1)sender:表示需要发射信号的对象。
(2)signal:表示需要发射的信号,该参数必须使用 SIGNAL()宏。
(3)receiver:表示接收信号的对象。
(4)method:表示与信号相关联的槽函数,这个参数也可以是信号,从而实现信号与信号的关联。该参数若是槽,需使用 SLOT()宏,若是信号需使用 SIGNAL 宏。
(4) type:用于指明信号和槽的关联方式,它决定了信号是立即传送到一个槽还是在稍后时间排队等待传送。关联方式使用枚举 Qt::ConnectionType 进行描述,下表为其取值及意义:


type取值说明

或者写成QObject::connect(......),或者connect(......)

connect有三种写法:

  • Qt4的写法
    QPushButton *button = new QPushButton("Quit");
    connect(button, SIGNAL(clicked()), &a, SLOT(quit()));

  • Qt5的写法
    QPushButton button("Quit");
    QObject::connect(&button, &QPushButton::clicked,&app, &QApplication::quit);** //即使quit有多个形参,也是这种形式**

  • Lambda


    Lambda表达式
Lambda示意图

函数对象参数:
(1)[],标识一个Lambda的开始,这部分必须存在,不能省略。函数对象参数是传递给编译器自动生成的函数对象类的构造函数的。函数对象参数只能 使用那些到定义Lambda为止时Lambda所在作用范围内可见的局部变量(包括Lambda所在类的this)。函数对象参数有以下形式:

  • 空。没有使用任何函数对象参数。
  • =。函数体内可以使用Lambda所在作用范围内所有可见的局部变量(包括Lambda所在类的this),并且是值传递方式(相当于编译器自动为我们按值传递了所有局部变量)。
  • &。函数体内可以使用Lambda所在作用范围内所有可见的局部变量(包括Lambda所在类的this),并且是引用传递方式(相当于编译器自动为我们按引用传递了所有局部变量)。
  • this。函数体内可以使用Lambda所在类中的成员变量。
  • a。将a按值进行传递。按值进行传递时,函数体内不能修改传递进来的a的拷贝,因为默认情况下函数是const的。要修改传递进来的a的拷贝,可以添加mutable修饰符。
  • &a。将a按引用进行传递。
  • a, &b。将a按值进行传递,b按引用进行传递。
  • =,&a, &b。除a和b按引用进行传递外,其他参数都按值进行传递。
  • &, a, b。除a和b按值进行传递外,其他参数都按引用进行传递。
      int m = 0, n = 0;
      [=] (int a) mutable { m = ++n + a; }(4);
      [&] (int a) { m = ++n + a; }(4);
      [=,&m] (int a) mutable { m = ++n + a; }(4);
      [&,m] (int a) mutable { m = ++n + a; }(4);
      [m,n] (int a) mutable { m = ++n + a; }(4);
      [&m,&n] (int a) { m = ++n + a; }(4);

(2)操作符重载函数参数;
标识重载的()操作符的参数,没有参数时,这部分可以省略。参数可以通过按值(如:(a,b))和按引用(如:(&a,&b))两种方式进行传递。

(3)可修改标示符;
mutable声明,这部分可以省略。按值传递函数对象参数时,加上mutable修饰符后,可以修改按值传递进来的拷贝(注意是能修改拷贝,而不是值本身)。

(4) 错误抛出标示符;
exception声明,这部分也可以省略。exception声明用于指定函数抛出的异常,如抛出整数类型的异常,可以使用throw(int)

(5) 函数返回值;
->返回值类型,标识函数返回值的类型,当返回值为void,或者函数体中只有一处return的地方(此时编译器可以自动推断出返回值类型)时,这部分可以省略。

(6)函数体;
{},标识函数的实现,这部分不能省略,但函数体可以为空。

2多种连接情况

信号与槽函数的连接有多种情况,如:一个信号连接一个槽、一个信号连接多个槽及多个信号连接同一个槽等等。


信号槽连接示意图
1. 一个信号连接一个槽
    connect(slider, &QSlider::valueChanged, spin_box, &QSpinBox::setValue);
2. 一个信号连接多个槽
    connect(slider, &QSlider::valueChanged, spin_box, &QSpinBox::setValue);
    connect(slider, &QSlider::valueChanged, this, &QWidget::showValue);
3. 多个信号连接同一个槽
    connect(push_button, &QPushButton::clicked, this, &QWidget::show);
    connect(tool_button, &QToolButton::clicked, this, &QWidget::show);
4. 一个信号连接另一个信号
    connect(push_button, &QPushlButton::clicked, this, &QWidget::buttonClicked);

class MyWidget : public QWidget 
{ 
public: 
    MyWidget(); 
... 
signals: 
    void aSignal(); 
... 
private: 
... 
    QPushButton *aButton; 
}; 
MyWidget::MyWidget() 
{ 
    aButton = new QPushButton( this ); 
    connect( aButton, SIGNAL(clicked()), SIGNAL(aSignal()) ); 
}

    在上面的构造函数中,MyWidget 创建了一个私有的按钮 aButton,按钮的单击事件产生的信号 clicked() 与另外一个信号 aSignal() 进行了关联。这样一来,当信号 clicked() 被发射时,信号 aSignal() 也接着被发射。当然,你也可以直接将单击事件与某个私有的槽函数相关联,然后在槽中发射 aSignal() 信号,这样的话似乎有点多余。

2-3断开信号和槽

取消函数为disconnect,参数形式与connect相对应。断开的情况可分为如下三种:

  1. 断开与某个对象相关联的任何对象
disconnect(&ma, 0, 0, 0);   断开与对象 ma 中的所有信号相关联的所有槽。
当我们在某个对象中定义了一个或者多个信号,这些信号与另外若干个对象中的槽相关联,如果我们要切断这些关联的话,就可以利用这个方法,非常之简洁
  1. 断开与某个特定信号的任何关联
disconnect( myObject, SIGNAL(mySignal()), 0, 0 ) 
或者myObject->disconnect( SIGNAL(mySignal()) )
  1. 断开两个对象之间的关联
disconnect( myObject, 0, myReceiver, 0 ) 
或者 myObject->disconnect(  myReceiver )

2-4总结

  1. 简单介绍了支持信号槽机制的元对象系统。
  2. 信号的标志符为signals,槽函数的标志符为slots,发射信号需要使用关键字emit。
  3. 通过函数connect连接信号与槽,通过disconnect取消信号与槽的连接。

参考资料
[1] https://www.ibm.com/developerworks/cn/linux/guitoolkit/qt/signal-slot/index.html
[2] https://blog.csdn.net/hyongilfmmm/article/details/83015045
[3] https://www.jianshu.com/p/d6fc0bb0689a
[4] https://blog.csdn.net/alzzw/article/details/99628409

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容

  • 这篇文档从使用到实现去讲信号-槽,适合小白到使用一两年的。对于Qt使用而言,信号-槽是我们津津乐道的一个功能,那我...
    开发呆呆阅读 2,144评论 0 0
  • 引自:https://blog.51cto.com/9291927/2070398 Qt高级——Qt信号槽机制源码...
    Magic11阅读 3,917评论 1 12
  • 信号和槽机制是 QT 的核心机制,所有从 QObject 或其子类 ( 例如 Qwidget) 派生的类都能够包含...
    土豆吞噬者阅读 817评论 0 0
  • 最近项目紧,人手不够,我也被拉过来做一些QT的项目,在次之前,我是一点都没接触过QT,不过项目要求,也就边上手边学...
    苏旋律阅读 544评论 0 0
  • 1、概述 信号槽是 Qt 框架引以为豪的机制之一。所谓信号槽,实际就是观察者模式。当某个事件发生之后,比如,按钮检...
    你的社交帐号昵阅读 45,274评论 0 9