You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

71 KiB

title date author top cover toc mathjax summary tags categories abbrlink reprintPolicy password coverImg img
QT 2022年4月28日 TianZD true true true false QT的安装教程,以及一些使用过程记录 [C++ QT] [C++ QT IDE] 8e7feb07 cc_by <nil> <nil> <nil>

[toc]

QT

QT 安装

中国科学技术大学:http://mirrors.ustc.edu.cn/qtproject/ 清华大学:https://mirrors.tuna.tsinghua.edu.cn/qt/ 北京理工大学:http://mirror.bit.edu.cn/qtproject/ 中国互联网络信息中心:https://mirrors.cnnic.cn/qt/

在这里插入图片描述

在这里插入图片描述

各模块意思 Qt Charts是二维图表模块,用于绘制柱状图、饼图、曲线图等常用二维图表。在制作一些需要绘制表格的软件的时候经常会用到,建议勾选 Qt Quick 3D模块初探,这个是技术预览勾不勾都可以,Quick 3D提供了用于基于Qt Quick创建3D内容或UI的高级API。提供了对现有Qt Quick场景图的扩展,以及对该扩展场景图的渲染器。不过还是在测试阶段,因为我可能会用到,所以我勾了。 Qt Data Visualization 是三维数据图表模块,用于数据的三维显示,如散点的三维空间分布、三维曲面等。这个如果你勾选了Qt Charts的话,这个基本也要用得到,建议勾选。 QT lottie animation 这个主要是用来实现复杂的动画效果,如果要用来制作动画或者制作游戏的话菜肴勾选。 Qt Purchasing、Qt WebEngine、Qt Network Auth(TP)等其他模块,括号里的 TP 表示技术预览。(技术预览就是就算你勾选了,也用不了,就是让你看看而已,哈哈哈) Qt Scritp(Deprecated)是脚本模块,括号里的“Deprecated”表示这是个已经过时的模块。 接下来是tool的选择


版权声明:本文为CSDN博主「流楚丶格念」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.csdn.net/weixin_45525272/article/details/113062352

解决Qt-至少需要一个有效且已启用的储存库 问题

blog.csdn.net

问题如图所示

image-20220218155724951

解决方法:

1、选择左下角的设置,进入如图界面,然后选择“临时存储库”

image-20220218155742235

2、手动添加临时储存库要定位一个储存有QT在线安装镜像的地址,这个地址可以从这里选择http://download.qt.io/static/mirrorlist/

在这个网站,显示了各国的qt镜像站点,中国有四个,随便选一个中科大的站点击HTTP会进入一个网络文件夹。

以此进入如下路径:/online/qtsdkrepository/windows_x86/root/qt/ 。然后把该路径添加到临时存储库中。然后就可以增删组件了。

最终路径如下所示

https://mirrors.tuna.tsinghua.edu.cn/qt/online/qtsdkrepository/windows_x86/root/qt/

Qt程序运行时CMD窗口

Linux 下都是带终端的. Windows 下面如果 pro 文件中 config+=console,就会带一个 cmd 窗口

Qt Creator中显示的汉字变为乱码

blog.csdn.net

问题是什么?

在学习Qt编程的过程中,大多数人都遇到过中文乱码的问题。总结起来有三类:

1. Qt Creator中显示的汉字变为乱码,编辑器上方有“Could not decode "..." with "UTF-8"-encoding. Editing not possible.”的错误提示。此时,出现乱码的文档是不可编辑的。如下图所示,“你好中文!”这5个中文字符变成了乱码:

image-20220218160928755

2. Qt Creator中显示的汉字正常,但编译的时候会出现“常量中有换行符”等一系列错误报警。其实,这也是文字编码的问题。如下图所示:

image-20220218160959524

3. 编译时未报错,但生成的程序中文乱码。如下图所示:

image-20220218161014007

其中,第3条是网上提问的最多的,几乎是所有使用MSVC的初学者都会碰到的问题。很多回答是针对Qt4版本的,Qt5中不可用。

为什么会出现这些问题?

在解决问题之前,字符编码知识是必需的。你要知道ASCII、GB2312、GBK、Unicode、UTF-8、UTF-16、BOM是怎么回事。此外,你还要明白源码字符集、执行字符集是什么。详细内容可以在网上搜索一下,俯拾即是。

1. Qt Creator的编辑器默认使用UTF-8(代码页65001)编码来读取文本文件。而Visual Studio保存文件时默认采用的是本地编码,对于简体中文的Windows操作系统,这个编码就是GB2312(代码页936)。如果使用Qt Creator读取由Visual Studio创建的文件,那么编辑器就会以UTF-8编码格式读取GB2312编码格式的文件,出现中文乱码,因为这两套编码系统对汉字编码是不同的。至于英文部分不会乱码,是因为UTF-8和GB2312在单字节字符部分是兼容的。

2. MSVC在编译时,会根据源代码文件有无BOM来定义源码字符集。如果有BOM,则按BOM解释识别编码;如果没有,则使用本地字符集,对于简体中文的Windows操作系统就是GB2312。那么,当MSVC遇到一个没有BOM的UTF-8编码的文件时,它通常会把文件看作GB2312的来处理。如果文件全是英文没有问题,但如果包含中文,编译器就会出现误读。这种情况下,Qt Creator编辑器是正常的。但对于MSVC编译器,原代码会被它认识成下图这个样子:

image-20220218161038546

这是我用EverEdit指定本地编码重读后的结果,可以看到汉字出错,末端的引号也没了。

在UTF-8中,一个中文字符(汉字或标点符号)占用3个字节,“你好中文!”这5个中文字符共占用15个字节;而在GB2312中,一个中文字符(汉字或标点符号)占用2个字节,这时,MSVC把UTF-8编码的15个字节加上后面1个字节的英文引号合成16个字节当作8个中文字符处理。之后,MSVC在这一行里直到末尾换行符出现都没有找到下一个引号,它以为你把字符串在这里敲回车换行了,于是报警称“常量中有换行符”,并引出一系列的错误。

不过,当以无BOM的UTF-8编码的字符串正好凑够偶数个字节时(比如偶数个汉字,或奇数个汉字加奇数个英文字母),编译器通常不会报警,因为它以为用GB2312编码读出的是正确的。

3. 不管源文件是何种编码,只要MSVC能够正确识别,就可以通过编译。但MSVC的执行字符集默认是本地字符集。对我们来说,它生成的可执行文件中的文字是GB2312编码的。而生成的Qt程序以UTF-8编码来识别GB2312编码的文字,对于“你好中文!”这几个字符,采用GB2312编码后再以UFT-8编码来读取,就会变成如下的乱码:

image-20220218161057717

当以无BOM的UTF-8编码的字符串正好凑够偶数个字节时(比如偶数个汉字,或奇数个汉字加奇数个英文字母),反而不会出现乱码。那是因为,编译器用GB2312编码读出的乱码本身就是UTF-8编码的,现在又用UTF-8解读,自然就正确了。这纯粹是歪打正着。

怎么解决这些问题?

首先,你要确定采用哪种源码字符集。你有两个选择:

1. 采用本地编码字符集(不推荐,跨平台时会比较麻烦,但在Visual Studio环境下配合Add-in工具编程比较方便);

2. 采用UTF-8编码字符集(推荐,适合跨平台)。

“采用本地编码字符集”方案,解决方法如下:

首先,要把项目中所有的头文件和源文件全都转换成GB2312编码保存。

1. 第1个问题:在Qt Creator中打开项目,点击左侧工具栏“项目”,在“编辑器”选项卡中把“默认编码”改成“GB2312”。如下图所示:

image-20220218161128126

话说回来,既然选择本地字符集,大致上是放弃跨平台了。与其用轻量级的Qt Creator,不如用Visual Studio作开发环境更好。

2. 第2个问题:“常量中有换行符”等一系列报警已不存在了。

3. 第3个问题:在字符串常量上加QStringLiteral宏或QString::fromLocal8Bit函数,如:

QString str = "你好中文!";

改为:

QString str = QStringLiteral("你好中文!");

或者:

QString str = QString::fromLocal8Bit("你好中文!");

不过,在这两种形式下,你都无法用tr方法来创建翻译了。

“采用UTF-8编码字符集”方案,解决方法如下:

注意:加上后,qdebug输出中文正常,cout输出中文乱码,窗口名称输出中文乱码

首先,要把项目中所有的头文件和源文件全都转换成UTF-8+BOM编码保存。

image-20220218172128013

1. 第1个问题不存在了。

2. 第2个问题也不存在了。

3. 第3个问题,你也可以用上个方案中的方法来解决,但有更好的方法。那就是要用到中文字符的头文件和源文件开头加上MSVC的一个宏:

#if _MSC_VER >= 1600
#pragma execution_character_set("utf-8")
#endif

这个宏告诉MSVC,执行字符集是UTF-8编码的,别瞎整成GB2312的!还有个好处,就是能用tr包中文,方便日后的翻译。

或者在pro文件中加入:

msvc {
    QMAKE_CFLAGS += /utf-8
    QMAKE_CXXFLAGS += /utf-8
}

最终效果如下:

image-20220218161244977

查看原网页: blog.csdn.net

回顾C++基础

类和对象

//新建类
class student{
public:
        string name;
        int age;
        void test()
};
void student::test(){//类外定义
    cout<<this->age<<endl;
}

    student stu1;//实例化对象
    stu1.age = 10;//通过点访问
    stu1.test();
    student *stu2 = new student;//在堆里定义,需要删除
    stu2->age = 11;//通过箭头访问
    stu2->test();

重载

重载:函数名相同,但是参数不同

构造函数和析构函数

析构函数:对象被删除或者生命周期结束时触发

构造函数 :对象被创建的时候触发、

student::student(){
    cout<<"hello,构造函数"<<endl;
}
student::~student(){
    cout<<"bye,析构函数"<<endl;
}

虚函数和纯虚函数

虚函数:有实际定义的,允许派生类对他进行覆盖替换,用virtual修饰

虚函数:没有实际定义的虚函数

QT工程基础

新建

项目文件结构

image-20220122165058047

.pro文件为项目文件

widget.ui文件为UI设计文件,双击进入UI界面设计

sources文件夹为代码文件夹

headers为库文件

.PRO文件解析

#-------------------------------------------------
#
# Project created by QtCreator 2022-01-22T16:46:31
#
#-------------------------------------------------

QT       += core gui    #添加core gui模块

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets #qt大于4时加入widgets模块

TARGET = QTDemo2    # 生成app的名称
TEMPLATE = app  ##编译产物的类型

# The following define makes your compiler emit warnings if you use
# any feature of Qt which as been marked as deprecated (the exact warnings
# depend on your compiler). Please consult the documentation of the
# deprecated API in order to know how to port your code away from it.
DEFINES += QT_DEPRECATED_WARNINGS   #定义的一个宏

# You can also make your code fail to compile if you use deprecated APIs.
# In order to do so, uncomment the following line.
# You can also select to disable deprecated APIs only up to a certain version of Qt.
#DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000    # disables all the APIs deprecated before Qt 6.0.0


SOURCES += \    #指定工程里有哪些CPP
        main.cpp \
        widget.cpp

HEADERS += \    #指定工程中有哪些文件
        widget.h

FORMS += \  #指定工程中有哪些ui文件
        widget.ui

widget.ui

image-20220122165312247

第一个qt工程

添加控件

image-20220122170231198

密码输入设置密文:

image-20220122171208294

image-20220122170248539

控件改名

为了分析代码方便,需要改名字

要通俗易懂

image-20220122171521061

信号和槽

信号:控件发出的特定的信号

:槽函数,把控件发出的信号绑定到特定的槽函数

关联信号和槽

  1. 自动关联:右键控件——转到槽,选择相应的信号
  2. 手动关联

自动关联

右键控件——转到槽,选择相应的信号

image-20220122171503322

选择相应的信号后,会自动进行:

  1. widget.hprivate slots下增加槽函数声明(注意,只能在private slots或者public slots下面)
private slots:
    void on_btnLogin_clicked();
  1. 自动在widget.cpp中增加函数定义(具体内容需要自己定义)
void Widget::on_btnLogin_clicked()
{
    
}
  1. 随后自己定义相应的槽函数即可
void Widget::on_btnLogin_clicked()
{
    qDebug("login");//在调试台输出login
}

手动关联

手动关联需要用到connect()函数

  1. 将自动关联步骤中的1、2、3手动实现,实现槽函数
void Widget::on_btnRegister_clicked()
{
    qDebug("register");
}
  1. widget.cpp中,添加关联函数

connect(ui->btnLogin,SIGNAL(clicked()),this,SLOT(on_btnRegister_clicked()));

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    connect(ui->btnRegister,SIGNAL(clicked()),this,SLOT(on_btnRegister_clicked()));//添加的关联函数
}

其中ui为指向整个窗口的指针,btnLogin为要关联的控件A,SIGNAL()函数中指定信号B,SLOT()函数中为槽函数C

当A发出B信号时,会触发槽函数C

添加图片

icon网址

图标之家

添加资源目录

image-20220122193234441

工程会出现Resources文件夹

将图片放入QT工程目录下

  1. 右键Resources文件夹下的文件,

image-20220122193716310

  1. 添加前缀

前缀设置成/,并保存

image-20220122193812191

  1. 点击添加并选择添加的图片

引用文件

  1. 在ui页面添加label控件(也支持按钮控件)
  2. 删掉控件中的文字,并右键——改变样式表——添加资源右侧的箭头——选择border-image——点击左侧——选择文件

image-20220122194046904

image-20220122194138559

UI界面布局

页面布局不会改变任何代码

概述

用途:防止因不同设备、不同缩放、不同分辨率导致布局错乱问题

QT布局样式:

  • 水平布局:对应1
  • 垂直布局:对应2
  • 栅格布局:对应3

image-20220122194911328

使用

选中两个控件,点击对应的布局按钮即可

弹簧控件Spacers

弹簧用来当缩放UI时,周边间距进行自动变化

image-20220122195329642

最终

image-20220122195549478

界面切换

创建一个新的界面

右键工程——添加新文件——QT——QT设计师界面类——widget——输入名字,这里为form

会自动在Headers文件夹下生成form.h头文件

image-20220122195907707

关联控件

修改widget.cpp文件,添加头文件和关联槽函数

#include "form.h"   //添加头文件
//省略
void Widget::on_btnLogin_clicked()
{
    //qDebug("register");
    Form *form1 = new Form;//创建一个Form对象
    form1->setGeometry(this->geometry());//设置form.ui的尺寸为当前ui的尺寸
    form1->show();//展示
}

关闭新界面

  1. 新UI界面控件改名
  2. 关联信号
  3. 设置关闭槽函数
void Form::on_pushButton_clicked()
{
    this->close();
}

获取界面输入

void Widget::on_btnLogin_clicked()
{
    //获取输入的参数
    QString name = ui->inputName->text();
    QString psd = ui->inputPassword->text();
    if(name == "admin" && psd == "123123"){
        Form *form1 = new Form;//创建一个Form对象
        form1->setGeometry(this->geometry());//设置form.ui的尺寸为当前ui的尺寸
        form1->show();//展示
    }
}

QT GUI设计

窗口置顶 与 取消置顶

void MainWindow::on_windowTopButton_clicked()
{
  if (m_flags == NULL)
  {
    m_flags = windowFlags();
    setWindowFlags(m_flags | Qt::WindowStaysOnTopHint);
    this->show();
  }
  else
  {
    m_flags = NULL;
    setWindowFlags(m_flags);
    this->show();
  }
}

回到顶部

全屏显示 与 退出全屏

首先,在QT中对于窗口显示常用的有这么几个方法可以调用:

showFullScreen() // Qt全屏显示函数
showMaximized() // Qt最大化显示函数
showMinimized() // Qt最小化显示函数
resize(x, y) // Qt固定尺寸显示函数
setMaximumSize(w, h) // Qt设置最大尺寸函数
setMinimumSize(w, h) // Qt设置最小尺寸函数

但是 showFullScreen()showNormal() 只对顶级窗口有效果,对子窗口无效。要将子窗口全屏显示可用以下方法:

  • 将要全屏的 Qt 子窗口调用 setWindowFlags(Qt::Window) 将其类型提升为顶级窗口模式,然后调用 showFullScreen() 函数将子窗口全屏显示。
  • 当然全屏后还要恢复正常,即调用 setWindowFlags(Qt::subwindow) 将子窗口设置为非顶级窗口,再调用 showNormal() 还原子窗口显示。
// 全屏显示:
VideoWidget->setWindowFlags(Qt::Window);
VideoWidget->showFullScreen();

// 退出全屏时:
VideoWidget->setWindowFlags(Qt::SubWindow);
VideoWidget->showNormal();

回到顶部

窗口的 "最大化\最小化\关闭" 按钮设置

1、在 QDialog 中添加 "最大化" 和 "最小化"按钮

在使用 QDialog 时,默认情况下只有 “这是什么” 和 “关闭” 按钮,但是我们习惯有最大化和最小化按钮。这里介绍如何在该模式下如何设置。

this->setWindowFlags(Qt::Dialog | Qt::WindowMinMaxButtonsHint | Qt::WindowCloseButtonHint);

其中 WindowMinMaxButtonsHint 设置的就是最大和最小按钮,但是只这样设置,关闭按钮就不可用了,因此需要再添加 WindowCloseButtonHint 一项。

2、只禁止最大化按钮

this->setWindowFlags(windowFlags() &~ Qt::WindowMaximizeButtonHint);

Qt::WindowMaximizeButtonHintWindowType 枚举变量,通过修改它可以选择禁止 "最小化" 或者 "关闭" 按钮。

回到顶部

禁止拖动窗口大小

this->setFixedSize(this->width(),this->height());

回到顶部

获取屏幕的宽度和高度

QApplication::desktop()->width();
QApplication::desktop()->height();

即得到屏幕分辨率,如 1024*768。

去掉菜单栏

    //隐藏菜单栏
    this->setWindowFlags(Qt::FramelessWindowHint |
                         Qt::WindowSystemMenuHint |
                         Qt::WindowMinMaxButtonsHint);

去掉以后,发现窗口无法移动,无法放大缩小关闭

添加自定义最小化、最大化、关闭按钮

mainwindow.h

protected :
    /**
     * @brief closeEvent重写closeEvent
     * @param event
     */
    void closeEvent(QCloseEvent *event);
private:
    //全屏显示标识
    bool fullScreenFlag = false;

mainwindow.cpp

/**
 * @brief MainWindow::closeEvent    重写closeEvent: 确认退出对话框
 * @param event
 */
void MainWindow::closeEvent(QCloseEvent *event)
{
    QMessageBox::StandardButton button;
    button=QMessageBox::question(this,tr("退出程序"),QString(tr("确认退出程序?")),QMessageBox::Yes|QMessageBox::No);
    if(button==QMessageBox::No)
    {
        event->ignore(); // 忽略退出信号,程序继续进行
    }
    else if(button==QMessageBox::Yes)
    {
        event->accept(); // 接受退出信号,程序退出
    }
}


/**
 * @brief MainWindow::on_closeBTN_clicked   自定义关闭按钮
 */
void MainWindow::on_closeBTN_clicked()
{
    this->close();
}

/**
 * @brief MainWindow::on_showMaximizedBTN_clicked   最大化按钮
 */
void MainWindow::on_showMaximizedBTN_clicked()
{
    if(!fullScreenFlag){
        this->showFullScreen();
        fullScreenFlag = true;
    }
    else{
         this->showNormal();
        fullScreenFlag = false;
    }
}
/**
 * @brief MainWindow::on_showMinBTN_clicked 最小化按钮
 */
void MainWindow::on_showMinBTN_clicked()
{
    this->showMinimized();
}

拖拽窗口移动

mainwindow.h

protected :
    /**
     * @brief 拖拽窗口
     */
    void mousePressEvent(QMouseEvent *event);
    void mouseMoveEvent(QMouseEvent *event);
    void mouseReleaseEvent(QMouseEvent *event);
private:
    Ui::MainWindow *ui;
    //拖拽窗口
    bool        m_bDrag;
    QPoint      mouseStartPoint;
    QPoint      windowTopLeftPoint;

mainwindow.cpp

/**
 * @brief MainWindow::mousePressEvent   拖拽窗口移动操作
 * @param event
 */
void MainWindow::mousePressEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        m_bDrag = true;
        //获得鼠标的初始位置
        mouseStartPoint = event->globalPos();
        //mouseStartPoint = event->pos();
        //获得窗口的初始位置
        windowTopLeftPoint = this->frameGeometry().topLeft();
    }
}

void MainWindow::mouseMoveEvent(QMouseEvent *event)
{
    if(m_bDrag)
    {
        //获得鼠标移动的距离
        QPoint distance = event->globalPos() - mouseStartPoint;
        //QPoint distance = event->pos() - mouseStartPoint;
        //改变窗口的位置
        this->move(windowTopLeftPoint + distance);
    }
}

void MainWindow::mouseReleaseEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        m_bDrag = false;
    }
}

设置空间透明

mode->setFlat(true);

//就是这句实现按钮透明的效果。

QT串口调试工具

网络编程、串口编程、操作GPIO

仿写

image-20220122215752070

串口调试助手UI界面

添加控件

  1. 设计UI界面大小800*480
  2. 添加数据接收框Plain Text Edit,并在属性设置区勾选read only
  3. 添加参数下拉选择输入框Combo Box和问题提示框label
  4. 添加发送框line Edit
  5. 添加按钮push button
  6. 添加广告框GroupBox+Label

image-20220122225909615

添加属性

  1. 双击波特率、数据位、停止位、校验位下拉输入框,添加属性

image-20220122230043510

  1. 通过属性currentIndex设置默认值

image-20220122231854565

控件改名

添加库支持

.pro文件中,增加serialport

QT       += core gui serialport

搜索串口并显示

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <QSerialPortInfo>//添加串口头文件

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    //获取串口号并显示
    QStringList serialPortNums;
    foreach (const QSerialPortInfo &info, QSerialPortInfo::availablePorts()) {
        serialPortNums<<info.portName();
    }
    ui->serialPortNum->addItems(serialPortNums);
}

Widget::~Widget()
{
    delete ui;
}

实现逻辑功能

创建串口对象

在widget.h中添加头文件,并声明串口对象指针

#include<QSerialPort>//引入串口头文件
public:
	QSerialPort *serialPort;//声明串口类指针

zai widget.cpp中创建对象

    serialPort = new QSerialPortthis;//创建串口对象

打开串口并初始化

  1. 关联打开串口控件的信号和槽函数

  2. 定义槽函数:定义串口数据——初始化

  3. 给当前串口对象赋值

void Widget::on_btnOpenSerial_clicked()//打开串口-点击信号对应的槽函数
{
    //声明变量
    QSerialPort::BaudRate baudRate;
    QSerialPort::DataBits dataBits;
    QSerialPort::StopBits stopBits;
    QSerialPort::Parity parity;
    //赋值,初始化
    //波特率
    if(ui->baudRate->currentText() == "4800"){
        baudRate = QSerialPort::Baud4800;
    }else if(ui->baudRate->currentText() == "9600"){
        baudRate = QSerialPort::Baud9600;
    }else if(ui->baudRate->currentText() == "115200"){
        baudRate = QSerialPort::Baud115200;
    }
    //数据位
    if(ui->dataBits->currentText()=="5"){
        dataBits = QSerialPort::Data5;
    }else if(ui->dataBits->currentText()=="6"){
        dataBits = QSerialPort::Data6;
    }else if(ui->dataBits->currentText()=="7"){
        dataBits = QSerialPort::Data7;
    }else if(ui->dataBits->currentText()=="8"){
        dataBits = QSerialPort::Data8;
    }
    //停止位
    if(ui->stopBits->currentText()=="1"){
        stopBits = QSerialPort::OneStop;
    }else if(ui->stopBits->currentText()=="1.5"){
        stopBits = QSerialPort::OneAndHalfStop;
    }else if(ui->stopBits->currentText()=="2"){
        stopBits = QSerialPort::TwoStop;
    }
    //校验位
    if(ui->parity->currentText()=="none"){
        parity = QSerialPort::NoParity;
    }

    //获取当前选择的串口号
    serialPort->setPortName(ui->serialPortNum->currentText());
    //设置串口参数
    serialPort->setBaudRate(baudRate);
    serialPort->setDataBits(dataBits);
    serialPort->setStopBits(stopBits);
    serialPort->setParity(parity);
}
  1. 判断串口是否打开成功
    //判断串口是否打开成功,并提示,需要添加<QMessgaeBox>头文件
    if(serialPort->open(QIODevice::ReadWrite) == true){
        QMessageBox::information(this,"提示","成功");
    }else {
        QMessageBox::critical(this,"提示","失败");
    }

image-20220123104230714

image-20220123104241542

关闭串口

  1. 关联信号和槽函数
  2. 定义逻辑
void Widget::on_btnCloseSerial_clicked()
{
    serialPort->close();
}

数据接收

  1. widget.hprivate slots下增加槽函数声明
private slots:
    void serialDataReadReady_Slot();//读数据槽函数声明
  1. widget.cpp中增加槽函数
//读数据槽函数定义
void Widget::serialDataReadReady_Slot(){
    QString buff;//暂存数据
    buff = QString(serialPort->readAll());//读取串口的数据
    ui->dataRCV->appendPlainText(buff);//将数据发送到接受区内
}
  1. widget.cpp中,添加关联函数
    //关联数据接收槽函数
    connect(serialPort,SIGNAL(readyRead()),this,SLOT(serialDataReadReady_Slot()));

数据发送

void Widget::on_btnSend_clicked()//数据发送槽函数
{
    serialPort->write(ui->dataInput->text().toLocal8Bit().data());
}

清空

void Widget::on_btnClear_clicked()//清空槽函数
{
    ui->dataRCV->clear();
}

QT程序打包和部署

为什么

  1. 把写好的程序给别人用

  2. 源码不能随便给别人

怎么做

release模式

  1. 把工程切换到release模式,然后编译

release模式,没有调试信息

debug模式,有很多调试信息

image-20220123112932311

  1. 找到release模式构建的文件夹,并进入release文件夹

image-20220123113146728

image-20220123113218111

Serial.exe文件就是所需的文件,但是之间无法打开,需要动态库

修改图标

图标之家

  1. 下载图标,格式为.ico,并拷贝到工程目录下,注意,不是编译目录

这里图标为:serial_ico.ico

image-20220123114501066

  1. 在.pro中增加图标,并编译
RC_ICONS = serial_ico.ico

编译可以看到左上角图标已经改变

image-20220123114615068

同时release目录下的exe文件的图标也已经改变

image-20220123114654768

封包

  1. 创建一个新的文件夹,不要有中文路径C:\code\QT\Serial_deploy

  2. 拷贝release下的exe文件到文件夹中

  3. 进入QT控制台,并进入新建的文件夹

直接搜索QT,应用下面的就是

image-20220123114858514

  1. 输入windeployqt Serial.exe命令后自动封包,结果如下

image-20220123115455018

  1. 双击里面的Serial.exe文件可以直接运行

程序打成安装包

Inno Setup软件

参考

QT网络编程-TCP通信

TcpServer

两个类:

  • QTcpServer
  • QTcpSocket

QTcpServer类

提供一个TCP基础服务类 继承自QObject

这个类用来接收到来的TCP连接,可以指定TCP端口或者用QTcpServer自己挑选一个端口,可以监听一个指定的地址或者所有的机器地址

调用listen()来监听所有的连接,每当一个新的客户端连接到服务端就会发射信号newConnection()

调用nextPendingConnection()来接受待处理的连接。返回一个连接的QTcpSocket(),我们可以用这个返回的套接字和客户端进行连接

如果有错误,serverError()返回错误的类型。调用errorString()来把错误打印出来。

当监听连接时候,可以调用serverAddress()serverPort()来返回服务端的地址和端口。

调用close()来关闭套接字,停止对连接的监听。

Header: #include
qmake: QT += network
Inherits: QObject
Inherited By: QSctpServer

成员函数

函数 用途
void close() 关闭服务,然后服务器讲不再监听任何连接
QString errorString()const 错误时候返回错误的字符串
bool hasPendingConnections()const 如果服务端有一个待处理的连接,就返回真,否则返回假
QTcpSocket* nextPendingConnection() 返回一个套接字来处理一个连接,这个套接字作为服务端的一个子对象,意味着当QTcpServer对象销毁时候,这个套接字也自动删除,当使用完后明确的删除这个套接字也好,这样可以避免内存浪费。当没有可处理的连接时候,这个函数返回0。
void incomingConnection(int socketDescriptor)[virtualprotected] 这个函数新建一个QTcpSocket套接字,建立套接字描述符,然后存储套接字在一个整形的待连接链表中。最后发射信号newConnection()
bool isListening()const 当服务端正在监听连接时候返回真,否则返回假
bool listen( const QHostAddress & address =QHostAddress::Any, quint16 port = 0 ) 告诉服务端监听所有来自地址为address端口为Port的连接,如果Port为0,那么会自动选择,如果address是QHostAddress::Any,那么服务端监听所有连接,成功返回1,否则返回0
int maxPendingConnections()const 返回最大允许连接数。默认是30
void setMaxPendingConnections(int numConnections) 设定待处理的连接最大数目为numConnections,当超过了最大连接数后,客户端仍旧可以连接服务端,但是服务端不在接受连接,操作系统会把这些链接保存在一个队列中。
QNetworkProxy proxy()const 返回这个套接字的网络代理层。
void setProxy(const QNetworkProxy & networkProxy) 设置这个套接字的网络代理层,进制使用代理时候,使用QNetworkProxy::NoProxy类型,例如server->setProxy(QNetworkProxy::NoProxy);
quint16serverPort()const serverAddress() 当服务端正在监听时候,返回服务端的端口和地址
bool waitForNewConnection(int msec=0,bool *timedOut=0) 最大等待msec毫秒或者等待一个新连接可用。如果一个连接可用,返回真,否则返回假。如果msec不等于0,那么超时将会被调用

QTcpSocket类

QTcpSocket 类提供一个TCP套接字

TCP是一个面向连接,可靠的的通信协议,非常适合于连续不断的数据传递

QTcpSocket 是QAbstractSocket类非常方便的一个子类,让你创建一个TCP连接和数据流交流。

Header: #include
qmake: QT += network
Inherits: QAbstractSocket
Inherited By: QSctpSocket and QSslSocket

成员函数

函数 用途
QTcpSocket::QTcpSocket ( QObject * parent = 0 ) 以UnconnectedState态创建一个QTcpSocket对象
QTcpSocket::~QTcpSocket () [virtual]‘ 析构函数,销毁对象
bool waitForConnected(int msecs = 30000) 等待,直到套接字被连接,最高为msecs毫秒。如果连接已经建立,这个函数返回true;否则返回false。在返回false的情况下,可以调用error()来确定错误的原因。
void connectToHost() 尝试连接给定端口上的主机名。如果查找成功,则发出hostFound(), QAbstractSocket进入ConnectingState。然后,它尝试连接到查找返回的一个或多个地址。最后,如果连接建立,QAbstractSocket进入ConnectedState并发出connected()。
[signal] void QAbstractSocket::connected() 这个信号在调用connectToHost()并成功建立连接之后发出。
bool disconnect() 断开对象发送器中的信号与对象接收器中的方法。如果连接成功断开,返回true;否则返回false。
void QAbstractSocket::disconnectFromHost() 试图关闭socket。如果有挂起的数据等待写入,QAbstractSocket将进入ClosingState并等待,直到所有的数据都被写入。最终,它将进入UnconnectedState并发出disconnected()信号。

TcpServer编写

创建ui界面

image-20220123152716836

引入network和两个头文件
QT       += core gui network
#include <QTcpServer>
#include <QTcpSocket>
声明对象
    //声明tcp所用的对象
    QTcpServer *tcpServer;
    QTcpSocket *tcpSocket;
创建对象widget.cpp
    //创建对象
    tcpServer = new QTcpServer(this);
    tcpSocket = new QTcpSocket(this);
编写槽函数
  1. 打开服务器时开启监听

调用listen()来监听所有的连接,每当一个新的客户端连接到服务端就会发射信号newConnection()

  1. 有新连接时创建tcpsocket,

调用nextPendingConnection()来接受待处理的连接。返回一个连接的QTcpSocket(),我们可以用这个返回的套接字和客户端进行连接

  1. 读取数据到接收框中

readyRead()readAll()

  1. 关闭服务器

close()

  1. 发送数据

write()

代码

widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpServer>
#include <QTcpSocket>

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

    //声明tcp所用的对象
    QTcpServer *tcpServer;
    QTcpSocket *tcpSocket;


private slots:
    void on_btnOpenServer_clicked();
    //2.有新连接时创建tcpsocket
    void newConnection_Slot();
    //3.读取就绪时读取数据
    void readyRead_SLOT();
    //4. 关闭服务器
    void on_btnCloseServer_clicked();
    //5. 发送数据
    void on_btnSendMessage_clicked();

private:
    Ui::Widget *ui;
};

#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    //创建对象
    tcpServer = new QTcpServer(this);
    tcpSocket = new QTcpSocket(this);

    //2.有新连接时创建tcpSocket
    connect(tcpServer,SIGNAL(newConnection()),this,SLOT(newConnection_Slot()));

}

Widget::~Widget()
{
    delete ui;
}

//打开服务器
void Widget::on_btnOpenServer_clicked()
{
    //1.打开监听,所有地址,端口由输入框设定
    tcpServer->listen(QHostAddress::Any,ui->tcpPort->text().toUInt());
}

//有新连接时创建tcpsocket
void Widget::newConnection_Slot(){
    //2.调用`nextPendingConnection()`来接受待处理的连接。返回一个连接的`QTcpSocket()`
    tcpSocket = tcpServer->nextPendingConnection();
    //3.有数据时,读取tcpSocket数据
    connect(tcpSocket,SIGNAL(readyRead()),this,SLOT(readyRead_SLOT()));
}

//3.有数据时,读取tcpSocket数据
void Widget::readyRead_SLOT(){
    QString buff;
    buff = tcpSocket->readAll();
    ui->messageRCV->appendPlainText(buff);
}

//4.关闭服务器
void Widget::on_btnCloseServer_clicked()
{
    //tcpServer->close();
    tcpSocket->close();
}

//5.发送数据
void Widget::on_btnSendMessage_clicked()
{
    //toLocal8Bit()转化为字符数组,data()转化为字符指针
    tcpSocket->write(ui->editMessageSend->text().toLocal8Bit().data());
}

测试

image-20220123161139876

TcpClient

QTcpSocket类

主要用到QTcpSocket类

QTcpClient编写

  1. 添加支持

.pro

QT       += core gui network #引入network
  1. 创建UI界面

image-20220123171112644

  1. 声明、创建QTcpSocket对象

创建QTcpSocket对象

  1. 打开连接

connectToHost()

  1. 连接成功时触发槽函数

connected()这个信号在调用connectToHost()并成功建立连接之后发出。

  1. 新数据时,槽函数

readyRead()有新数据到来时触发

  1. 发送数据

write()

toLocal8Bit()转化为字符数组,data()转化为字符指针

  1. 关闭连接

close()

代码

TcpClient.pro

略去

widget.h
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
//只需要引入socket
#include <QTcpSocket>

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
    //1.socket对象
    QTcpSocket *tcpSocket;

private slots:
    void on_btnConnect_clicked();
    //3.连接成功槽函数
    void connected_Slot();
    //4.新数据到来时触发的槽函数
    void readyRead_Slot();
    //5.发送数据
    void on_btnSend_clicked();
    //6.关闭连接
    void on_btnClose_clicked();

private:
    Ui::Widget *ui;
};

#endif // WIDGET_H
widget.cpp
#include "widget.h"
#include "ui_widget.h"
#include <QMessageBox>

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    //1.socket对象
    tcpSocket = new QTcpSocket(this);

}

Widget::~Widget()
{
    delete ui;
}

void Widget::on_btnConnect_clicked()
{
    //2.打开连接
    tcpSocket->connectToHost(ui->editIpInput->text(),ui->editPortInput->text().toUInt());
    if(tcpSocket->waitForConnected(1000)){
        QMessageBox::information(this,"提示","连接成功");
    }else{
        QMessageBox::critical(this,"提示","连接失败");
    }
    //3.连接成功时连接函数
    //3.`connected()`这个信号在调用connectToHost()并成功建立连接之后发出。
    connect(tcpSocket,SIGNAL(connected()),this,SLOT(connected_Slot()));
}
//3.连接成功槽函数
void Widget::connected_Slot(){
    //4.新数据时,连接函数
    //`readyRead()`有新数据到来时触发
    connect(tcpSocket,SIGNAL(readyRead()),this,SLOT(readyRead_Slot()));
}
//4.新数据时槽函数
//发送数据到接收框
void Widget::readyRead_Slot(){
    QString buff;
    buff = tcpSocket->readAll();
    ui->pteMessageRCV->appendPlainText(buff);
}

//5.发送数据槽函数
void Widget::on_btnSend_clicked()
{
    //toLocal8Bit()转化为字符数组,data()转化为字符指针
    tcpSocket->write(ui->editMessageSend->text().toLocal8Bit().data());
}


//6.关闭连接槽函数
void Widget::on_btnClose_clicked()
{
    tcpSocket->close();
    //tcpSocket->disconnectFromHost();
    QMessageBox::information(this,"提示","断开连接");
}

测试

image-20220123170825078

image-20220123172226132

QT网络编程-UDP通信

概述

UDP不分客户端和服务端

只需要使用QUdpSocket

QUdpSocket类

判断是否有数据等待读取

bool QUdpSocket::hasPendingDatagrams() const

如果至少有一个数据报正在等待读取,则返回true;否则返回false。

返回数据报的大小

qint64 QUdpSocket::pendingDatagramSize() const

返回第一个挂起的UDP数据报的大小。如果没有可用的数据报,这个函数返回-1。

读取数据

qint64 QUdpSocket::readDatagram(char **data*, qint64 *maxSize*, QHostAddress **address* = Q_NULLPTR, quint16 **port* = Q_NULLPTR)

接收不大于maxSize字节的数据报,并将其存储在数据中。发送者的主机地址和端口存储在address和port中(除非指针为0)。

成功时返回数据报的大小;否则返回1。

如果maxSize太小,则数据报的其余部分将丢失。为了避免数据丢失,在尝试读取挂起的数据报之前,调用pendingDatagramSize()来确定它的大小。如果maxSize为0,该数据报将被丢弃。

绑定端口

bool QAbstractSocket::bind(const QHostAddress &address, quint16 port = 0, BindMode mode = DefaultForPlatform)

使用BindMode模式在端口端口上绑定到地址。

将这个套接字绑定到地址地址和端口端口。

对于UDP套接字,绑定后,当UDP数据报到达指定的地址和端口时,就会发出信号QUdpSocket::readyRead()。因此,这个函数对编写UDP服务器很有用。

对于TCP套接字,这个函数可以用来指定出连接使用哪个接口,这在有多个网络接口的情况下很有用。

默认情况下,套接字使用DefaultForPlatform BindMode绑定。如果不指定端口,则选择随机端口。

成功时,函数返回true,套接字进入BoundState;否则返回false。

bool QAbstractSocket::bind(quint16 port = 0, BindMode mode = DefaultForPlatform)

这是一个重载函数。 绑定到QHostAddress:任何端口端口,使用BindMode模式。 默认情况下,套接字使用DefaultForPlatform BindMode绑定。如果不指定端口,则选择随机端口。

发送数据

qint64 QUdpSocket::writeDatagram(const char *data, qint64 size, const QHostAddress &address, quint16 port)

将数据报以数据大小发送到主机地址在端口的地址端口。返回成功时发送的字节数;否则返回1。

数据报总是写成一个块。数据报的最大大小高度依赖于平台,但可以低至8192字节。如果数据报太大,这个函数将返回-1,error()将返回DatagramTooLargeError。

发送大于512字节的数据报通常是不建议的,因为即使它们成功发送,它们也可能在到达最终目的地之前被IP层分片。

警告:在一个连接的UDP套接字上调用这个函数可能会导致错误和没有发送数据包。如果您正在使用已连接的套接字,请使用write()发送数据报。

qint64 QUdpSocket::writeDatagram(const QNetworkDatagram &datagram)

这是一个重载函数。

使用那里设置的网络接口和跳数限制,将数据报数据报发送到包含在数据报中的主机地址和端口号。如果没有设置目的地址和端口号,这个函数将发送到传递给connectToHost()的地址。

如果目的地址是IPv6,范围id非空,但与数据报中的接口索引不同,则操作系统将选择发送哪个接口是未定义的。

如果函数成功,则返回发送的字节数;如果遇到错误,则返回-1。

警告:在一个连接的UDP套接字上调用这个函数可能会导致错误和没有发送数据包。如果您正在使用已连接的套接字,请使用write()发送数据报。

qint64 QUdpSocket::writeDatagram(const QByteArray &datagram, const QHostAddress &host, quint16 port)

这是一个重载函数。 将数据报数据报发送到主机地址主机和在端口端口。 如果函数成功,则返回发送的字节数;如果遇到错误,则返回-1。

其他

bool QUdpSocket::joinMulticastGroup(const QHostAddress&*groupAddress*)

加入操作系统选择的缺省接口上的groupAddress指定的组播组。套接字必须处于BoundState状态,否则将发生错误。

注意,如果你试图加入一个IPv4组,你的套接字不能使用IPv6(或双模式,使用QHostAddress::Any)绑定。你必须使用QHostAddress::AnyIPv4代替。

如果成功,此函数返回true;否则返回false并设置相应的socket错误。

bool QUdpSocket::joinMulticastGroup(const QHostAddress &*groupAddress*, const QNetworkInterface &*iface*)

这是一个重载函数。

加入该接口的组播组地址groupAddress。

bool QUdpSocket::leaveMulticastGroup(const QHostAddress &*groupAddress*)

将groupAddress指定的组播组保留在操作系统选择的缺省接口上。套接字必须处于BoundState状态,否则将发生错误。

如果成功,此函数返回true;否则返回false并设置相应的socket错误。

bool QUdpSocket::leaveMulticastGroup(const QHostAddress &*groupAddress*, const QNetworkInterface &*iface*)

这是一个重载函数。

离开接口上的groupAddress指定的组播组。

QNetworkInterface QUdpSocket::multicastInterface() const

返回组播数据报的出接口的接口。这对应于IPv4套接字的IP_MULTICAST_IF套接字选项和IPv6套接字的IPV6_MULTICAST_IF套接字选项。如果之前没有设置接口,这个函数返回一个无效的QNetworkInterface。套接字必须处于BoundState,否则返回无效的QNetworkInterface。

QNetworkDatagram QUdpSocket::receiveDatagram(qint64 *maxSize* = -1)

接收一个不大于maxSize字节的数据报,并在QNetworkDatagram对象中返回它,以及发送方的主机地址和端口。如果可能,此函数还将尝试确定数据报的目的地址、端口和接收时的跳数。

失败时,返回一个报告无效的QNetworkDatagram。

如果maxSize太小,则数据报的其余部分将丢失。如果maxSize为0,该数据报将被丢弃。如果maxSize为-1(默认值),这个函数将尝试读取整个数据报。

void QUdpSocket::setMulticastInterface(const QNetworkInterface &*iface*)

设置组播数据报的出接口为接口interface。这对应于IPv4套接字的IP_MULTICAST_IF套接字选项和IPv6套接字的IPV6_MULTICAST_IF套接字选项。套接字必须处于BoundState,否则这个函数什么都不做。

实例

编写界面

image-20220123173536182

引入支持文件

qmake:network

header:\#include <QUdpSocket>

代码

  • 创建QUdpSocket对象

  • 打开连接

bind()

readyRead()

  • 读取数据

hasPendDatagrams()

hasPendDatagramsSize()

readDatagrams()

  • 发送数据

writeDatagram()

  • 关闭
widget.h
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
//添加头文件Qudpsocket
#include <QUdpSocket>

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

    //1.创建对象
    QUdpSocket *udpSocket;

private slots:
    //2.打开连接槽函数
    void on_btnOpen_clicked();
    //3.readyRead槽函数
    void readyRead();
    //4.发送
    void on_btnSend_clicked();
    //关闭
    void on_btnClose_clicked();

private:
    Ui::Widget *ui;
};

#endif // WIDGET_H

widget.cpp
#include "widget.h"
#include "ui_widget.h"
#include <QMessageBox>
#include <QHostAddress>

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    //1.创建对象
    udpSocket = new QUdpSocket(this);

}

Widget::~Widget()
{
    delete ui;
}

void Widget::on_btnOpen_clicked()
{
    //2.打开连接
    //判断是否成功
    if(udpSocket->bind(ui->editLocalPort->text().toUInt()) == true){
        QMessageBox::information(this,"提示","成功");
    }else{
        QMessageBox::critical(this,"提示","失败");
    };
    //3.连接成功触发函数
    connect(udpSocket,SIGNAL(readyRead()),this,SLOT(readyRead()));
}

void Widget::readyRead(){
    //3.槽函数,读取数据
    while(udpSocket->hasPendingDatagrams()){    //判断是否有数据
        QByteArray ary;
        ary.resize(udpSocket->pendingDatagramSize());
        udpSocket->readDatagram(ary.data(),ary.size());

        QString buff;
        buff = ary.data();
        ui->editMessageRCV->appendPlainText(buff);
    }
}

void Widget::on_btnSend_clicked()
{
    //4.发送数据
    //数据
    QString sendBuff;
    sendBuff = ui->editMessageSend->text();
    //ip
    QHostAddress addr;
    addr.setAddress(ui->editTargetIP->text());
    //端口
    quint16 port;
    port = ui->editTargetPort->text().toUInt();
    //发送数据
    udpSocket->writeDatagram(sendBuff.toLocal8Bit().data(),sendBuff.length(),addr,port);
}

void Widget::on_btnClose_clicked()
{
    //5.关闭
    udpSocket->close();
}

测试

image-20220123182754858

QT定时器

		QTimer *timer = new QTimer(this);
      connect(timer, SIGNAL(timeout()), this, SLOT(update()));
      timer->start(1000)

Qt QMessageBox用法详解

c.biancheng.net

QMessageBox 是 Qt 框架中常用的一个类,可以生成各式各样、各种用途的消息对话框,如图 1 所示。

图 1 QMessageBox消息对话框

很多 GUI 程序都会用到消息对话框,且很多场景中使用的消息对话框是类似的,唯一的区别只是提示信息不同。为了提高程序员的开发效率,避免重复地“造轮子”,Qt 开发者设计好了几种通用的 QMessageBox 消息对话框,需要时可以直接使用。

通用的QMessageBox消息框

Qt 提供了 6 种通用的 QMessageBox 消息对话框,通过调用 QMessageBox 类中的 6 个静态成员方法,可以直接在项目中使用它们。

1) information消息对话框

information 对话框常用于给用户提示一些关键的信息,它的外观如下图所示:

图 2 information 消息对话框

在项目中使用 information 消息对话框,直接调用 QMessageBox 类中的 information() 静态成员方法即可,该方法的语法格式如下:

StandardButton QMessageBox::information(QWidget *parent,
                                        const QString &title,
                                        const QString &text,
                                        StandardButtons buttons = Ok,
                                        StandardButton defaultButton = NoButton)

各个参数的含义是:

  • parent:指定消息对话框的父窗口,消息提示框会作为一个独立的窗口显示在父窗口的前面。消息提示框从弹出到关闭的整个过程中,用户无法操作父窗口,更不能删除父窗口;
  • title:指定消息对话框的标题,即图 2 中的 Titile;
  • text:指定消息对话框的具体内容,即图 2 中的 text;
  • buttons:指定消息对话框中包含的按钮。默认情况下,消息对话框只包含一个按钮,即图 2 中显示的 "OK" 按钮。根据需要,我们可以用|按位或运算符在消息对话框中设置多个按钮,例如 QMessageBox::Ok|QMessageBox::Cancel
  • defaultButton:指定 Enter 回车键对应的按钮,用户按下回车键时就等同于按下此按钮。注意,defaultButton 参数的值必须是 buttons 中包含的按钮,当然也可以不手动指定,QMessageBox 会自动从 buttons 中选择合适的按钮作为 defaultButton 的值。

information() 函数会返回用户按下的按钮。StandardButton 是 QMessageBox 类中定义的枚举类型,每个枚举值代表一种按钮。StandardButton 类型中的值有很多,下表给大家罗列了几个常用的:

表 1 QMessageBox::StandardButton 枚举类型值

枚举值 含 义
QMessageBox::Ok 标有 "OK" 字样的按钮,通常用来表示用户接受或同意提示框中显示的信息。
QMessageBox::Open 标有 "Open" 字样的按钮。
QMessageBox::Save 标有 "Save" 字样的按钮。
QMessageBox::Cancel 标有 "Cancel" 字样的按钮。点击此按钮,通常表示用户拒绝接受提示框中显示的信息。
QMessageBox::Close 标有 "Close" 字样的按钮。
QMessageBox::Discard 标有 "Discard" 或者 "Don't Save" 字样的按钮,取决于运行平台。
QMessageBox::Apply 标有 "Apply" 字样的按钮。
QMessageBox::Reset 标有 "Reset" 字样的按钮。
QMessageBox::Yes 标有 "Yes" 字样的按钮。
QMessageBox::No 标有 "No" 字样的按钮。

例如,使用 information() 函数实现图 2 所示的对话框,实现代码为:

QMessageBox::StandardButton result = QMessageBox::information(&widget, "Title","text");

其中,widget 是我们创建好的 QWidget 窗口,创建好的 information 对话框会显示在 widget 窗口的前面。通过用 result 接收 information() 函数的返回值,我们可以得知用户选择的是哪个按钮。

2) critical消息对话框

critical 消息对话框常用于给用户提示“操作错误”或“运行失败”的信息,它的外观如下图所示:

图 3 critical 消息对话框

项目中使用 critical 消息对话框,直接调用 QMessageBox 类提供的 critical() 静态成员方法即可,该方法的语法格式为:

StandardButton QMessageBox::critical(QWidget *parent,
                                     const QString &title,
                                     const QString &text,
                                     StandardButtons buttons = Ok,
                                     StandardButton defaultButton = NoButton)

各个参数的含义以及返回值的含义,都与 information() 函数相同,这里不再重复赘述。

例如,使用 critical() 函数实现图 3 所示的对话框,实现代码为:

QMessageBox::StandardButton result=QMessageBox::critical(&widget, "Title","text");

其中,widget 是我们创建好的 QWidget 窗口,创建好的 critical 对话框会显示在 widget 窗口的前面。

3) question消息对话框

question 对话框常用于向用户提出问题并接收用户的答案,它的外观如下图所示:

  • 图 4 question消息对话框

项目中使用 question 对话框,可以直接调用 QMessageBox 类的 question() 静态成员方法,该方法的语法格式为:

StandardButton QMessageBox::question(QWidget *parent,
                                     const QString &title,
                                     const QString &text,
                                     StandardButtons buttons = StandardButtons( Yes | No ),
                                     StandardButton defaultButton = NoButton)

各个参数的含义以及返回值的含义,都与 information() 函数相同。

例如,使用 question() 函数实现图 4 所示的对话框,实现代码为:

QMessageBox::StandardButton result\=QMessageBox::question(&widget, "Title","text");

其中,widget 是我们创建好的 QWidget 窗口,创建好的 question 对话框会显示在 widget 窗口的前面。

4) warning消息对话框

warining 对话框常用于向用户显示一些警告信息,它的外观如下图所示:

图 5 warning消息对话框

项目中使用 warning 对话框,可以直接调用 QMessageBox 类的 warning() 静态成员方法,该方法的语法格式为:

StandardButton QMessageBox::warning(QWidget *parent,
                                    const QString &title,
                                    const QString &text,
                                    StandardButtons buttons = Ok,
                                    StandardButton defaultButton = NoButton)

各个参数的含义以及返回值的含义,都与 information() 函数相同。

例如,使用 warning() 函数实现图 5 所示的对话框,实现代码为:

QMessageBox::StandardButton result=QMessageBox::warning(&widget, "Title","text");

其中,widget 是我们创建好的 QWidget 窗口,创建好的 warning 对话框会显示在 widget 窗口的前面。

5) about和aboutQt对话框

about 对话框常常作为介绍某个产品或某项功能的临时窗口,它的外观如下图所示:

图 6 about消息对话框

注意,about 对话框没有固定的图标,它显示的图标可能来自父窗口、包含父窗口的顶层窗口等,也可能使用和 information 对话框相同的图标。

项目中使用 about 对话框,直接调用 QMessageBox 类提供的 about() 静态成员方法即可,该方法的语法格式如下:

void QMessageBox::about(QWidget *parent, const QString &title, const QString &text)

各个参数的含义和与 information() 函数相同。和前面的几种对话框不同,about对话框中只包含一个默认的 Ok 按钮,且 about() 函数没有返回值。

aboutQt 可以看做是 about 对话框的一个具体实例,它只能显示 Qt 的介绍信息,如下图所示:

图 7 aboutQt对话框

项目中使用 aboutQt 对话框,直接调用 QMessageBox 类提供的 aboutQt() 静态成员方法即可,该函数的语法格式如下:

void QMessageBox::aboutQt(QWidget *parent, const QString &title = QString())

我们只能设置 aboutQt 对话框的 parent 父窗口和 title 标题,不能自定义它的内容。所以在实际场景中,aboutQt() 对话框很少使用。

自定义QMessageBox对话框

以上 6 种通用的 QMessageBox 对话框,界面上的图片无法修改,按钮上的文字也无法修改(例如无法将 OK、No 改成中文)。如果想修改它们,就需要自定义一个 QMessageBox 对话框。

QMessageBox对话框的创建

程序中创建 QMessageBox 对象,必须先引入<QMessageBox>头文件。QMessageBox 类提供了两个构造函数,分别是:

QMessageBox::QMessageBox(QWidget *parent = Q_NULLPTR)
QMessageBox::QMessageBox(Icon icon,
                         const QString &title,
                         const QString &text,
                         StandardButtons buttons = NoButton,
                         QWidget *parent = Q_NULLPTR,
                         Qt::WindowFlags f = Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint)

第一个构造函数可以创建一个“空白”对话框,即对话框中不包含任何文本和按钮。当然,通过调用 QMessageBox 类提供的成员方法,可以向“空白”对话框中添加各种元素(图标、文本、按钮等)。

第二个构造函数中,各个参数的含义是:

  • icon:指定对话框中的图片。Icon 是 QMessageBox 类中定义的枚举类型,内部包含 QMessageBox::NoIcon、QMessageBox::Question、QMessageBox::Information、QMessageBox::Warning、QMessageBox::Critical 几个值,分别表示:不指定图片、question对话框的图片(图 4)、information对话框的图片(图 2)、warning对话框的图片(图 5)、critical对话框的图片(图 3)。
  • title:指定对话框的标题;
  • text:指定对话框中显示的文本信息;
  • buttons:指定对话框中包含的的按钮,可选值如表 1 所示。
  • parent:指定对话框的父窗口;
  • f:指定对话框的属性。WindowFlags 是 Qt 提供的枚举类型,内部包含的值有很多,有的用来指定对话框的用途(比如 Qt::Dialog 表示对话框窗口),有的用来指定对话框的外观(比如 MSWindowsFixedSizeDialogHint 表示给对话框添加一个细的边框)

举个简单的例子:

 #include <QApplication>
#include <QMessageBox>
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    //创建 QMessageBox 类对象
    QMessageBox MyBox(QMessageBox::Question,"Title","text",QMessageBox::Yes|QMessageBox::No);
    //使 MyBox 对话框显示
    MyBox.exec();
    return a.exec();
}

程序中创建了一个 MyBox 对话框,通过调用 QMessageBox 提供的 exec() 方法,可以使 MyBox 对话框弹出。运行程序可以发现,MyBox 对话框的外观和图 4 的 question 对话框完全一样。

QMessageBox对话框的使用

QMessageBox 类提供了很多功能实用的成员方法,方便我们快速地制作出实际场景需要的对话框。

下表给大家罗列了常用的一些 QMessageBox 类成员方法:

表 2 QMessageBox 常用成员方法

成员方法 功 能
void QMessageBox::setWindowTitle(const QString &title) 设置对话框的标题。
void setText(const QString &text) 设置对话框中要显示的文本。
void setIconPixmap(const QPixmap &pixmap) 设置对话框中使用的图片。
QAbstractButton *QMessageBox::clickedButton() const 返回用户点击的按钮。
QPushButton *QMessageBox::addButton(const QString &text, ButtonRole role) 向对话框中添加按钮,text 为按钮的文本,role 是 QMessageBox::ButtonRole 枚举类型的变量,用于描述按钮扮演的角色,它的可选值有 QMessageBox::AcceptRole(同 OK 按钮)、QMessageBox::RejectRole(同 Cancel 按钮)等。
int QMessageBox::exec() 使当前对话框弹出,除非用户关闭对话框,否则对话框将一直存在。此外,当对话框中使用的都是 Qt 提供的按钮时,该方法可以监听用户点击的是哪个按钮,并将该按钮对应的枚举值返回;如果对话框中包含自定义按钮,需要借助 clickedButton() 方法确定用户点击的按钮。

举个简单的例子:

#include <QApplication>
#include <QMessageBox>
#include <QPushButton>
#include <QDebug>
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    QMessageBox MBox;
    MBox.setWindowTitle("QMessageBox自定义对话框");
    MBox.setText("这是一个自定义的对话框");
    MBox.setIconPixmap(QPixmap("C:\\Users\\xiexuewu\\Desktop\\icon_c.png"));
    QPushButton *agreeBut = MBox.addButton("同意", QMessageBox::AcceptRole);
    MBox.exec();
    if (MBox.clickedButton() == (QAbstractButton*)agreeBut) {
        //在 Qt Creator 的输出窗口中输出指定字符串
        qDebug() << "用户点击了同意按钮";
    }
    return a.exec();
}

程序运行结果如图 8 所示,点击“同意”按钮后,我们会在 Qt Creator 的输出窗口中看到“用户点击了同意按钮”。

图 8 自定义的 QMessageBox 对话框

QMessageBox的信号和槽

操作 QMessageBox 对话框,最常用的信号函数是 buttonClicked() 函数,最常用的槽函数是 exec() 函数,它们的语法格式和功能如下表所示。

表 3 QMessageBox信号和槽

信号函数 功 能
void QMessageBox::buttonClicked(QAbstractButton *button) 当用户点击对话框中的某个按钮时,会触发此信号函数,该函数会将用户点击的按钮作为参数传递给槽函数。
槽函数 功 能
int QMessageBox::exec() 弹出对话框,直到用户手动关闭对话框,此对话框将一直存在。

举个简单的例子:

//main.cpp
#include <QApplication>
#include <QWidget>
#include <QMessageBox>
#include <QPushButton>
#include <QAbstractButton>
QPushButton* agreeBut;
QPushButton* disagreeBut;
class MyWidget:public QWidget{
    Q_OBJECT
public slots:
    void buttonClicked(QAbstractButton * butClicked);
};

void MyWidget::buttonClicked(QAbstractButton * butClicked){
    if(butClicked == (QAbstractButton*)disagreeBut){
        this->close();
    }
}

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    //创建主窗口
    MyWidget myWidget;
    myWidget.setWindowTitle("主窗口");
    myWidget.resize(400,300);

    //创建消息框
    QMessageBox MyBox(QMessageBox::Question,"","");
    MyBox.setParent(&myWidget);
    //设置消息框的属性为对话框,它会是一个独立的窗口
    MyBox.setWindowFlags(Qt::Dialog);
    MyBox.setWindowTitle("协议");
    MyBox.setText("使用本产品,请您严格遵守xxx规定!");
    //自定义两个按钮
    agreeBut = MyBox.addButton("同意", QMessageBox::AcceptRole);
    disagreeBut = MyBox.addButton("拒绝", QMessageBox::RejectRole);

    myWidget.show();
    //添加信号和槽,监听用户点击的按钮,如果用户拒绝,则主窗口随之关闭。
    QObject::connect(&MyBox,&QMessageBox::buttonClicked,&myWidget,&MyWidget::buttonClicked);
    MyBox.exec();

    return a.exec();
}
//MyWidget类的定义应该放到 .h 文件中,本例中将其写到 main.cpp 中,程序最后需要添加 #include "当前源文件名.moc" 语句,否则无法通过编译。
#include "main.moc"

程序执行结果为:

查看原网页: c.biancheng.net