1. Qt消息模型

(1)Qt封装了具体操作系统的消息机制

(2)Qt遵循经典的GUI消息驱动事件模型

2. 信号与槽

(1)Qt中定义了与系统消息相关的概念

  ①信号(Signal):由操作系统产生的消息

  ②槽(Slot):程序中的消息处理函数

  ③连接(Connect):将系统消息绑定到消息处理函数(映射规则)

(2)Qt中的消息处理机制

(3)Qt的核心——QObject::connect函数

  ①函数原型

bool connect(const QObject* sender,     //发送对象
const char* signal, //消息名
const QObject* receiver, //接收对象
const char* method, //接收者成员函数(消息处理函数)
Qt::ConnectionType type = Qt::AutoConnection);

  ②在Qt中,消息用字符串进行描述,method是成员函数的名字(也是字符串类型)

  ③connect函数在消息名和处理函数之间建立映射

(4)Qt中的“新”关键字

  ①SIGNAL:用于指定消息名

  ②SLOT:用于指定消息处理函数名

  ③Q_OBJECT:所有自定义槽的类必须在类声明的开始处加上Q_OBJECT

  ④slots:用于在类中声明消息处理函数

【编程实验】初探信号和槽

#include <QApplication>
#include <QPushButton> int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QPushButton b; //顶级组件 b.setText("Click me to quit!");
b.show(); //将按钮b的clicked消息映射到a的quit函数
QObject::connect(&b, SIGNAL(clicked()), &a, SLOT(quit())); return a.exec();
}

3. 自定义槽

(1)只有QObject的子类才能自定义槽

(2)定义槽的类必须在声明的最开始处使用Q_OBJECT

(3)类中声明槽时需要使用slots关键字

(4)与所处理的信号在函数签名上必须一致(函数签名,即相同参数及类型、返回值

(5)SIGNAL和SLOT所指定的名称中,可以包含参数类型,但不能包含具体的参数名

【编程实验】为计算器实例添加消息处理函数

//main.cpp

#include <QApplication>
#include "QCalculatorUI.h" int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QCalculatorUI* cal = QCalculatorUI::NewInstance();
int ret = -; if(cal != NULL)
{
cal->show();
ret = a.exec(); delete cal;
} return ret;
}

//QCalculatorUI.h

#ifndef _QCALCULATORUI_H_
#define _QCALCULATORUI_H_ #include <QWidget>
#include <QLineEdit>
#include <QPushButton> class QCalculatorUI : public QWidget
{
//要自定义信号和槽,必须在最开始的这里添加Q_OBJECT
Q_OBJECT
private:
QLineEdit* m_edit;
QPushButton* m_buttons[]; //二阶构造法:当new一个QLineEdit和一些按钮时可能会失败,所以采用二阶构造
QCalculatorUI(); //第1阶——先隐藏构造函数
bool construct();//第2阶 private slots: //声明槽时得加slots
void onButtonClicked(); public:
static QCalculatorUI* NewInstance();
void show();
~QCalculatorUI();
}; #endif //_QCALCULATORUI_H_

//QCalculatorUI.cpp

#include "QCalculatorUI.h"
#include <QDebug> QCalculatorUI::QCalculatorUI(): QWidget(NULL, Qt::WindowCloseButtonHint)
{ } bool QCalculatorUI::construct()
{
bool ret = true; const char* btnText[] =
{
"", "", "", "+", "(",
"", "", "", "-", ")",
"", "", "", "*", "←",
"", ".", "=", "/", "C",
}; m_edit = new QLineEdit(this);//le的生命期由父组件来管理
if( m_edit != NULL)
{
m_edit->move(, );
m_edit->resize(, );
m_edit->setReadOnly(true); //设置编辑框的只读属性
}
else
{
ret = false;
return ret;
} for(int i = ; (i < ) && ret; i++)
{
for(int j = ; (j< ) && ret; j++)
{ m_buttons[i * + j] = new QPushButton(this);//按钮的生命期由父组件来管理
if (m_buttons[i * + j] != NULL)
{
m_buttons[i * + j]->resize(, );
m_buttons[i * + j]->move( + j * , + i * );
m_buttons[i * + j]->setText(btnText[i * + j]); //消息映射
//1.消息名(信号)要加SIGNAL关键字,消息处理函数(槽):用SLOT关键字
//2.信号和槽的函数签名必须一致,即都是无参的函数,返回值void
connect(m_buttons[i*+j], SIGNAL(clicked()), this, SLOT(onButtonClicked()));
}
else
{
ret = false;
}
}
} return ret;
} QCalculatorUI* QCalculatorUI::NewInstance()
{
QCalculatorUI* ret = new QCalculatorUI(); if((ret == NULL) || !ret->construct())
{
delete ret;//删除半成品
ret = NULL;
} return ret;
} void QCalculatorUI::show()
{
QWidget::show();
setFixedSize(width(), height());
} void QCalculatorUI::onButtonClicked()
{
//sender是QObject类的,用于表示消息的发送者
QPushButton* btn = (QPushButton*)sender(); qDebug() << "onButtonClicked()";
qDebug() << btn->text(); } QCalculatorUI::~QCalculatorUI()
{ }

(6)小贴士:解决经典问题——Object::connect:No such slot…

  ①检查类是否继承于QObject

  ②检查类声明的开始处是否添加Q_OBJECT

  ③检查是否使用slots关键字进行槽声明

  ④检查槽的名称是否拼写错误

  ⑤重新执行qmake

4. 小结

(1)信号与槽是Qt中的核心机制

(2)不同的Qt对象可以通过信号和槽进行通信

(3)只有QObject的子类才能自定义信号和槽

(4)使用信号和槽的类必须在声明的最开始处使用Q_OBJECT

(5)信号与处理函数在函数签名上必须一致。

第10课 初探 Qt 中的消息处理的更多相关文章

  1. 第38课 Qt中的事件处理(上)

    1. GUI程序原理回顾 (1)图形界面应用程序的消息处理模型 (2)思考:操作系统发送的消息如何转变为Qt信号 2. Qt中的事件处理 (1)Qt平台将系统产生的消息转换为Qt事件 ①Qt事件是一个 ...

  2. 第30课 Qt中的文本编辑组件

    1. 3种常用的文本编辑组件的比较 单行文本支持 多行文本支持 自定义格式支持 富文本支持 QLineEdit (单行文本编辑组件) Yes No No No QPlainTextEdit (多行普通 ...

  3. 第11课 Qt中的字符串类

    1. 历史遗留问题和解决方案 (1)历史遗留问题 ①C语言不支持真正意义上的字符串 ②C语言用字符数组和一组函数实现字符串操作 ③C语言不支持自定义类型,因此无法获得字符串类型 (2)解决方案 ①从C ...

  4. 第7课 Qt中的坐标系统

    1. 坐标系统 (1)GUI操作系统都有特定的坐标系统 (2)图形界面程序在坐标系统中进行窗口和部件的定位 (3)定位类型 ①顶级窗口部件的定位 ②窗口内部件的定位 ③窗口部件的大小设置 (4)QWi ...

  5. 第47课 Qt中的调色板

    1. QPalette类 (1)QPalette类提供了绘制QWidget组件的不同状态所使用的颜色. (2)QPalette对象包含了3个状态的颜色描述 ①激活颜色组(Active):组件获得焦点使 ...

  6. 第39课 Qt中的事件处理(下)

    1. 事件的传递过程 (1)操作系统检测到用户动作时,会产生一条系统消息,该消息被发送到Qt应用程序 (2)Qt应用程序收到系统消息后,将其转化为一个对应的QEvent事件对象,并调用QObject: ...

  7. 第32课 Qt中的文件操作

    1. Qt的中IO操作 (1)Qt中IO操作的处理方式 ①Qt通过统一的接口简化了文件和外部设备的操作方式 ②Qt中的文件被看作一种特殊的外部设备 ③Qt中的文件操作与外部设备的操作相同 (2)IO操 ...

  8. 信号和槽:Qt中最差劲的创造

    不要被这个标题唬住了,实际上我是非常认可Qt的.在C++实现的开源产品中没有哪一个的API风格比得上Qt,拥有高度一致性,符合常识,符合直觉,几乎不用学就可以直接上手.或许是由于我们摆脱不了马太效应的 ...

  9. Qt中 QString 和int, char等的“相互”转换

    转载:http://blog.csdn.net/ei__nino/article/details/7297791 Qt中 int ,float ,double转换为QString 有两种方法 1.使用 ...

随机推荐

  1. BZOJ 1566 【NOI2009】 管道取珠

    题目链接:管道取珠 这道题思路还是很巧妙的. 一开始我看着那个平方不知所措……看了题解后发现,这种问题有一类巧妙的转化.我们可以看成两个人来玩这个游戏,那么答案就是第二个人的每个方案在第一个人的所有方 ...

  2. Eclipse开发Android应用 找不到平板

    1.驱动安装正确2.平板的连接方式正确,不要用大容量存储/sd卡模式这个设置在4.3上很难找呀.设置->存储->点右上角的菜单 3.打开USB调试.4.上述问题都检查后,在eclipse里 ...

  3. maven 工程mybatis自动生成实体类

    generatorConfig.xml <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE ge ...

  4. HTML字符转码

    以下是HTML特殊字符的编码表: 标记 编码 实际名称 ™ ™ € € Space   ! ! " " " # # $ $ % % & & & ' ...

  5. iOS UI-自动布局(AutoLayout)

    // // ViewController.m // IOS_0115_AutoLayout // // Created by ma c on 16/1/15. // Copyright (c) 201 ...

  6. 001——数组(一)数组知识及foreach函数应用

    <?php /**数组(一)数组知识及foreach函数应用*/ /*数组:在一个变量中,存储一个或多个值,每一个元素都有一个访问ID * * */ /* * //索引型数组 $arr=arra ...

  7. 创建Vue.js对象:我的第一个Vue.js输出信息

    <!DOCTYPE html><html><head><meta charset=”utf-8″><title>Vue第一条信息</t ...

  8. 更新.xsd后,rdlc 数据源更新不了

  9. CentOS安装Zabbix

    rpm -i http://repo.zabbix.com/zabbix/3.4/rhel/7/x86_64/zabbix-release-3.4-2.el7.noarch.rpm使用MySQL数据库 ...

  10. Oozie_04总结一下workflowf的运行流程【20161116】

    4.1 bin/oozie ..... 提交任务 [hadoop@hadoop01 oozie-4.0.0-cdh5.3.6]$ bin/oozie job -oozie http://hadoop0 ...