赞
踩
元对象系统是一个基于标准C++的扩展,为QT提供了信号与槽机制、实时类型信息、动态属性系统。
元对象系统的三个基本条件:类必须继承自QObject、类声明Q_OBJECT宏(默认私有)、元对象编译器moc。
信号与槽机制是QT的核心机制,信号与槽是一种高级接口,应用于对象之间的通信。信号和槽是QT自行定义的一种通信机制,它独立于标准的C/C++语言,要正确的处理信号和槽,必须借助一个称为moc(Meta-Object-Compiler),也就是"元对象编译器"。
它为高层次的事件处理自动生成所需要的必要代码。QT程序在交由标准编译器编译之前,先要使用moc分析C++源文件。如果moc发现在一个类头文件中包含了函Q_OBJECT,则会生成以moc_className.cpp(自定义类名)的.cpp文件。这个源文件中包含了Q_OBJECT宏的实现代码。新的文件同样将进入编译系统,与源文件一起参与编译。构建生成的.o文件包含与moc生成的.cpp文件中。
产生的历史背景:
GUI用户界面中,当用户操作一个窗口部件时,需要其他窗口部件响应,传统方式经常使用callback(回调机制)来实现。所谓回调即事先将函数指针作为一个参数传递给另一个函数,然后在函数处理过程中适当地方调用函数。
回调机制有两个缺陷:类型不安全,不能保证调用过程中使用正确的参数,强耦合,处理函数必须知道调用哪个回调函数。
QT的信号与槽机制:
QT的信号与槽机制是类型安全的,松耦合,更灵活,更方便。
信号与槽(Signal & Slot)是QT编程的基础,也是QT的一大创新。因为有了信号与槽机制的编程,在QT中处理界面的各个组件的交互操作时变得更加直观个简单。
**信号(Signal)**就是在特定情况下被发射的事件,例如PushButton最常见的信号就是鼠标单击时发射的clicked()信号。发生信号使用QT的emit关键字。QT的signals关键字指出进入了信号的声明区,随后即可声明自己的信号。
**槽(Slot)**就是对信号响应的函数。槽就是一个函数,与一般的C++函数是一样的,可以声明在类的任何部分(public、private or protected),可以具有任何参数,也可以被直接调用。
槽函数与一般的函数不同的是:槽函数可以与一个信号关联(connect),当信号被发射时, 关联的槽函数被自动执行。
信号与槽链接方式
mainwindow.h
#ifndef MAINWINDOW_H #define MAINWINDOW_H #include <QMainWindow> namespace Ui { class MainWindow; } class MainWindow : public QMainWindow { Q_OBJECT public: explicit MainWindow(QWidget *parent = 0); ~MainWindow(); //自定义信号 //信号的声明 //信号不用实现也不能实现 signals: void Comeon(QString& str); private slots: //同样可以手动添加槽方法-或者在设计模式中添加槽方法 void on_WorldBtn_clicked(); void on_CNBtn_clicked(); void on_BJBtn_clicked(); void startSend(QString& str); private: Ui::MainWindow *ui; }; #endif // MAINWINDOW_H
mainwindow.cpp
#include "mainwindow.h" #include "ui_mainwindow.h" #include<qdebug.h> MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow) { ui->setupUi(this); //链接信号与槽 //默认是自动关联 //将信号和处理这个信号的槽方法相连接起来,connect只负责将它们两个链接在一起 //至于谁发的,就无所谓了。 connect(this,SIGNAL(Comeon(QString&)),this,SLOT(startSend(QString&))); //或者 //connect(this,&MainWindow::Comeon,this,&MainWindow::startSend); } MainWindow::~MainWindow() { delete ui; } void MainWindow::on_WorldBtn_clicked() { //发射信号 QString str = "Hello World!"; emit Comeon(str); } void MainWindow::on_CNBtn_clicked() { QString str = "Hello China!"; emit Comeon(str); } void MainWindow::on_BJBtn_clicked() { QString str = "Hello BeiJing!"; emit Comeon(str); } void MainWindow::startSend(QString &str) { //打印调试信息 qDebug()<<str; }
一个信号对应多个槽方法。可以多个槽响应一个信号。
示例:
connect(this,SIGNAL(Comeon(QString&)),this,SLOT(startSend(QString&)));
connect(this,SIGNAL(Comeon(QString&)),this,SLOT(HelloCN()));
connect(this,SIGNAL(Comeon(QString&)),this,SLOT(HelloCHP()));
**注意:**一般情况下多个槽方法的参数个数保持一致,**但是,**有的时候,可以不使用信号传过来的这个参数,所以槽方法的参数个数可以比信号传递的参数个数少,但是,不能比信号传递过来的参数数量多。
错误示范:
connect(this,SIGNAL(Comeon(QString&)),this,SLOT(HelloCN(QString&,int)));
链接失败。
多个信号链接一个槽方法。进一步反应了信号与槽机制的灵活性。
示例:
connect(this,SIGNAL(Comeon(QString&)),this,SLOT(HelloCHP()));
connect(this,SIGNAL(Comeon2()),this,SLOT(HelloCHP()));
connect(this,SIGNAL(Comeon3()),this,SLOT(HelloCHP()));
connect(this,SIGNAL(Comeon4()),this,SLOT(HelloCHP()));
参数设置见上《信号与槽链接(一对多)》。
当一个信号发射时,发射另一个信号。
connect(this,SIGNAL(Comeon4()),this,SIGNAL(Comeon3()));
断开一个信号和一个与它相链接的槽方法。
disconnect,参数同connect
disconnect(this,SIGNAL(Comeon(QString&)),this,SLOT(startSend(QString&)));
断开所有与该信号相连接的槽方法。
与上面不同的是,后两个参数置为0,所有的对象,和所有的槽方法。
disconnect(this,SIGNAL(Comeon(QString&)),0,0),;
断开指定对象的所有信号与的指定对象的所有槽方法
qDebug()<<"断开this所有信号与this的所有槽方法的链接";
disconnect(this,0,this,0);
disconnect(this,0,0,0);//断开this所有相关的对象的槽方法的链接
也可以通过connect函数的返回值来断开某个链接。
//获取返回值
......
private:
QMetaObject::Connection m_res;//返回值类型
......
m_res = connect(this,SIGNAL(Comeon(QString&)),this,SLOT(startSend(QString&)));//获取返回值
......
disconnect(m_res);//断开链接
总结-信号与槽机制的优越性:
在标准C++中,为了保证封装性,我们经常声明一个私有变量,然后声明两个共有函数例如set,get来对这个变量进行操作。
同理,在QT中我们可以使用宏**Q_PROPERTY()**来实现这些,函数可以使用QObject::property()和QObject::setProperty()。
在使用的时候,我们不用知道变量的所在类的任何细节,只需要知道名字即可。
Q_PROPERTY(type name
(READ getFunction [WRITE setFunction] |
MEMBER memberName [(READ getFunction | WRITE setFunction)])
[RESET resetFunction]
[NOTIFY notifySignal]
[REVISION int]
[DESIGNABLE bool]
[SCRIPTABLE bool]
[STORED bool]
[USER bool]
[CONSTANT]
[FINAL])
示例:
......
Q_PROPERTY(QString mask READ mask WRITE setMask NOTIFY maskChanged)
......
QObject* obj = myPc;
qDebug()<<obj->property("mask").toString();
qDebug()<<obj->property("mask").toString();
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。