Qt信号与槽原理

信号和槽是用于对象之间的通信的,这是Qt的核心。为此Qt引入了一些关键字,他们是slots、signals、emit,这些都不是C++关键字,是Qt特有的,这些关键字会被Qt的moc转换为标准的C++语句。Qt的部件类中有一些已经定义好了的信号和槽,通常的作法是子类化部件类,然后添加自已的信号和槽。

因为信号和槽与函数相似,所以通常把信号称为信号函数,槽称为槽函数。

信号和槽原理

C++虽然是面象对象的语言,但程序的具体实现代码仍然是由函数来实现的,因此所谓的对象之间的通信,从程序设计语言语法角度来看就是函数调用的问题,只不过是某个对象的成员函数调用另一个对象的成员函数而已,本文从语法角度讲解信号和槽的原理,这样更容易理解信号和槽的实现原理。注:信号和槽其实是观察者模式的一种实现,有兴趣的读者可以参阅《设计模式》课程。

函数调用的几种形式

见图2-1,假设函数f需要g的处理结果,有以下几种处理方式

最简单的方式就是直接调用函数g,但这种方式有一个明显的缺点,必须知道函数g的名称“g”以及函数g的参数类型。但是若f只需要g的处理结果就可以了,而g的处理结果不一定需要函数g来完成,它也可以是x、y或其他函数来完成,那么这种直接调用函数的方式就无法胜任了,因为系统不知道用户会使用哪个函数来完成这种处理结果,也就是系统不知道调用的函数名究竟是g、x或其他名称。

另一种方式就是回调函数,即在函数f中使用一个指向函数的指针去调用需要的函数,这样就可以调用任意名称的函数(只要函数类型与指针相同即可),此时只要是完成了函数g功能的函数都可以作为函数f的结果被调用,这样就不会被函数名称所限制。比如

Qt使用的信号和槽机制

注意:信号和槽不是C++标准代码,因此这些代码需要使用Qt的moc进行重新编译。 其基本思想如下:

创建一个信号,其中创建信号需要一些规则。

当需要调用外部函数时,发送一个信号,

此时与该信号相关联的槽便会被调用,槽其实就是一个函数,当然要使函数成为槽是有一定规则的。槽与信号的关联需要由程序员来完成。

在Qt中,信号和槽都需要位于类之中。

示例(具体实现时Qt引入了一些新的关键字):

创建信号和槽

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

1、信号需符合以下规则

信号使用signals关键字声明,在其后面有一个冒号“:”,在其前面不能有public、private、protected访问控制符,信号默认是public的。

信号只需像函数那样声明即可,其中可以有参数,参数的主要作用是用于和槽的通信,这就像普通函数的参数传递规则一样。信号虽然像函数,但是对他的调用方式不一样,信号需要使用emit关键字发送。

信号只需声明,不能对其进行定义,信号是由moc自动生成的。

信号的返回值只能是void类型的。

2、声明槽需符合以下规则

声明槽需要使用slots关键字,在其后面有一个冒号“:”,且槽需使用public、private、protected访问控制符之一。

槽就是一个普通的函数,可以像使用普通函数一样进行使用,槽与普通函数的主要区别是,槽可以与信号关联。

3、发射信号需符合以下规则:

发射信号需要使用emit关键字,注意,在emit后面不需要冒号。

emit发射的信号使用的语法与调用普通函数相同,比如有一个信号为void f(int),则发送的语法为:emit f(3);

当信号被发射时,与其相关联的槽函数会被调用(注意:信号和槽需要使用QObject::connect函数进行关联之后,发射信号后才会调用相关联的槽函数)。

注意:因为信号位于类之中,因此发射信号的位置需要位于该类的成员函数中或该类能见到信号的标识符的位置。

信号和槽的关系

槽的参数的类型需要与信号参数的类型相对应,

槽的参数不能多余信号的参数,因为若槽的参数更多,则多余的参数不能接收到信号传递过来的值,若在槽中使用了这些多余的无值的参数,就会产生错误。

若信号的参数多余槽的参数,则多余的参数将被忽略。

一个信号可以与多个槽关联,多个信号也可以与同一个槽关联,信号也可以关联到另一个信号上。

若一个信号关联到多个槽时,则发射信号时,槽函数按照关联的顺序依次执行。

若信号连接到另一个信号,则当第一个信号发射时,会立即发射第二个信号。

因Qt在其类库中预定义了很多信号和槽,因此在Qt中可以仅使用Qt类库中预定义的信号和槽,也可以只使用Qt类库中预定义的信号而使用自已的槽,也可以使用Qt类库中预定义的槽来响应自已定义的信号,当然,槽和信号也都可以使用自定义的。

示例2.10:创建信号和槽

//头文件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

//源文件m.cpp的内容

#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; }


信号和槽的关联(连接)

信号和槽使用QObject类中的成员函数connect进行关联,该函数有多个重载版本,如下所示。






6、形式3与形式1的区别

形式1的SINGAL和SLOT宏实际是把该宏的参数转换为字符串,当信号和槽相关联时,使用的是字符串进行匹配,因此,信号和槽的参数类型的名字必须在字符串意义上相同,所以信号和槽无法使用兼容类型的参数,也因此也不能使用typedef或namespace的类型,虽然他们的实际类型相同,但由于字符串名字不同,从而无法使用形式1。

形式3的信号和槽函数的参数类型不需完全一致,可以进行隐式转换。形式3还支持typedef和命名空间。

形式3以指针的形式指定信号和槽函数,不需再使用SIGNAL()和SLOT宏。

形式3的槽函数可以不使用slots关键字声明,任意的成员函数都可以是槽函数。形式1的槽函数必须使用slots修饰。

形式1的槽函数不受private的限制,也就是说即使槽是private的,仍可通过信号调用该槽函数,而形式3则在使用connect时就会发生错误。

当信号或槽函数有重载的形式时,使用形式3可能会产生二义性错误,此时可使用函数指针的形式指定信号或槽函数,或者使用形式1,比如

示例2.11:信号和槽的关联

//头文件m.h的内容

#ifndef M_H

#define M_H

#include<QObject>

#include <iostream>

using namespace std;

class A:public QObject{ Q_OBJECT

    signals:void s();    void s(int,int);    void s1();    void s2(int); };

class B:public QObject{  Q_OBJECT

public slots:

        void x(){cout<<"x"<<endl;}        void y(int i,int j){cout<<"y="<<i<<j<<endl;}

        void z(int){cout<<"zi"<<endl;}      void z1(float){cout<<"z1f"<<endl;}

public:

        void z2(){  //注意,该函数未使用slots声明。

            cout<<"z2"<<endl; }

private slots:            //私有槽

void z3(){    cout<<"z3"<<endl;        } };

#endif // M_H

//源文件m.cpp的内容

#include "m.h"

int main(int argc, char *argv[]){    A ma;    B mb;

    QObject::connect(&ma,SIGNAL(s()),&mb,SLOT(x()));  //形式1

    emit ma.s();    //输出x

    QObject::connect(&ma,&A::s1,&mb,&B::x);  //形式3

    emit ma.s1();  //输出x

  // QObject::connect(&ma,&A::s1,&mb,&B::y);  //错误,槽参数的数量多余信号参数的数量

//类型转换

//关联失败,形式1不支持类型转换

  //QObject::connect(&ma,SIGNAL(s2(int)),&mb,SLOT(z1(float))); 

    typedef int T;

//关联失败,对于形式1,类型T和int在字符串形式上并不相同。

//QObject::connect(&ma,SIGNAL(s2(T)),&mb,SLOT(z(int)));

QObject::connect(&ma,&A::s2,&mb,&B::z1);  //正确,形式3支持隐式类型转换

    emit ma.s2(2);  //输出z1f

//槽函数与slots

//关联失败,形式1的槽必须使用slots声明。

//QObject::connect(&ma,SIGNAL(s1()),&mb,SLOT(z2()));

    QObject::connect(&ma,&A::s1,&mb,&B::z2);  //正确,形式3的槽不需使用slots声明。

    emit ma.s1();    /*输出x,z2,注意:在之前s1和x的关联并未断开,此时信号s1同时与槽x和z2关联。*/

//访问控制符

    QObject::connect(&ma,SIGNAL(s1()),&mb,SLOT(z3())); //正确,形式1的槽不受访问控制符限制。

    emit ma.s1();  //输出x,z2,z3,因为此时s1与多个槽相关联。

    //QObject::connect(&ma,&A::s1,&mb,&B::z3);//错误,z3是私有的,形式3会受访问控制符的限制。

//函数重载,形式2和形式4的使用见正文

    return 0; }

作者:黄邦勇帅(原名:黄勇)

Linux、C/C++技术交流群:整理了一些个人觉得比较好的学习书籍、大厂面试题、热门技术教学视频资料共享在里面,(包括C/C++,Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK等等.)有需要的可以自行添加哦!


以上不足的地方欢迎指出讨论,觉得不错的朋友,希望能得到您的点赞支持

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

推荐阅读更多精彩内容

  • 一、问题 学习Qt有一段时间了,信号槽用的也是666,可是对信号槽的机制还是一知半解,总觉着不是那么得劲儿,万一哪...
    行走的代码阅读 1,107评论 0 5
  • 本文使用Qt 5.12.6 + MinGW7.3.0.64+win10环境+qtbase源码 我们来看一下以下几个...
    zypper阅读 2,795评论 0 1
  • 写在前面 在界面搭建时,我们需要很多触发事件。比如,鼠标右击,我们希望弹出对话框;点击退出的按钮,界面就会关闭或者...
    锅盖666阅读 582评论 0 0
  • 引自:https://blog.51cto.com/9291927/2070398 Qt高级——Qt信号槽机制源码...
    Magic11阅读 3,949评论 1 12
  • 推荐指数: 6.0 书籍主旨关键词:特权、焦点、注意力、语言联想、情景联想 观点: 1.统计学现在叫数据分析,社会...
    Jenaral阅读 5,714评论 0 5