QT文件IO

七、文件IO

  1. QFileDialog文件对话框

与QMessageBox一样,QFileDialog也继承了QDialog类,直接使用静态成员函数弹窗,弹窗的结果(选择的文件路径)通过函数返回值返回。

// 获取一个打开或保存的文件路径
// 参数1:父对象
// 参数2:即windwTitle属性(标题)
// 参数3:在那个目录中打开,默认值表示项目的工作目录
// 参数4:文件格式过滤器
// 返回值:选择的文件路径,如果选择失败,返回空字符
QString	getOpenFileName(
        QWidget * parent = 0, const QString & caption = QString(), const QString & dir = QString(), const QString & filter = QString())
QString QFileDialog::getSaveFileName(
        QWidget * parent = 0, const QString & caption = QString(), const QString & dir = QString(), const QString & filter = QString())
[static]

需要注意以下,QFileDialog只是一个窗口类,本身不具备任何IO的能力。

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :QDialog(parent),ui(new Ui::Dialog)
{
    ui->setupUi(this);connect(ui->pushButtonOpen,SIGNAL(clicked()),this,SLOT(btnClickedSlot()));connect(ui->pushButtonSave,SIGNAL(clicked()),this,SLOT(btnClickedSlot()));}Dialog::~Dialog()
{delete ui;
}void Dialog::btnClickedSlot()
{if(ui->pushButtonOpen == sender()){QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";QString path = QFileDialog::getOpenFileName(this,"打开","D:/",filter);if(path != ""){
            readPath = path;
            ui->textBrowserOpen->append(path);}else if(readPath == ""){
            QMessageBox::warning(this,"提示","请选择打开的文件");}}else if(ui->pushButtonSave == sender()){QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";QString path = QFileDialog::getSaveFileName(this,"保存","D:/",filter);if(path != ""){
            writePath = path;
            ui->textBrowserSave->append(path);}else if(writePath == ""){
            QMessageBox::warning(this,"提示","请选择保存的文件");}}else if(ui->pushButtonCopy == sender()){}
}

2、QFileInfo文件信息类

只需要创建出对象后,通过各种成员函数直接获取文件信息。

// 构造函数
// 参数为文件路径,如果文件非法,仍然可以创建出QFileInfo对象
QFileInfo::​QFileInfo(const QString & file)

// 判断文件是否存在
// 如果存在则返回true。否则返回false
bool QFileInfo::​exists() const

// 返回基础名称,不包含后缀
QString QFileInfo::​baseName() const

// 获取文件大小
// 返回为文件大小
qint64 QFileInfo::​size() const

// 返回文件可读性,true可读、false不可读
bool QFileInfo::​isReadable() const

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog)
{
    ui->setupUi(this);    connect(ui->pushButtonOpen,SIGNAL(clicked()),this,
            SLOT(btnClickedSlot()));
    connect(ui->pushButtonSave,SIGNAL(clicked()),this,
            SLOT(btnClickedSlot()));}Dialog::~Dialog()
{
    delete ui;
}void Dialog::printFileInfo()
{
    // 创建文件对象
    QFileInfo fileInfo(readPath);    // 判断文件是否存在
    if(!fileInfo.exists())
    {
        QMessageBox::warning(this,"提示","文件路径无效");
        return;
    }
    QString text = fileInfo.baseName();
    text.prepend("文件名称:");
    ui->textBrowserOpen->append(text);    qint64 size = fileInfo.size();
    text = QString::number(size);
    text.prepend("文件大小:").append("字节");
    ui->textBrowserOpen->append(text);    if(fileInfo.isReadable())
    {
        ui->textBrowserOpen->append("文件可读");
    }
    else
    {
        ui->textBrowserOpen->append("文件不可读");
    }
}void Dialog::btnClickedSlot()
{
    if(ui->pushButtonOpen == sender())
    {
        QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";
        QString path = QFileDialog::getOpenFileName(this,"打开","D:/",filter);
        if(path != "")
        {
            readPath = path;
            ui->textBrowserOpen->append(path);
            printFileInfo(); // 文件读取成功后输出信息
        }
        else if(readPath == "")
        {
            QMessageBox::warning(this,"提示","请选择打开的文件");
        }    }
    else if(ui->pushButtonSave == sender())
    {
        QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";
        QString path = QFileDialog::getSaveFileName(this,"保存","D:/",filter);
        if(path != "")
        {
            writePath = path;
            ui->textBrowserSave->append(path);
        }
        else if(writePath == "")
        {
            QMessageBox::warning(this,"提示","请选择保存的文件");
        }
    }
    else if(ui->pushButtonCopy == sender())
    {    }}

3、QFile文件读写类

在Qt中所有IO类都继承自QIODevice类,QIODevice类中规定了最基础的IO相关接口,这些接口虽然在不同的派生类中可能实现有所不同,但调用方式一致。

// 构造函数
// 参数为文件路径,如果是非法路径,也能创建出QFIle对象,但是不能正常IO输入输出操作。
QFile::​QFile(const QString & name)

// 判断QFile对应的文件是否存在
bool QFile::​exists() const

// 打开数据流
// 参数为打开模式、只读模式、只写模式、读写模式
bool QIODevice::​open(OpenMode mode)[virtual]

// 构造函数,QByteArray是qt中的常用数组
// 构造一个空子节数组
QByteArray::​QByteArray()

// 是否读取到文件末尾
bool QIODevice::​atEnd() const[virtual]

// 读取数据
// 参数:每次读取的大小
// 返回值QByteArray字符数组
QByteArray QIODevice::​read(qint64 maxSize)

// 写入数据
// 要写入的数据
// 返回值,本次写入的大小,返回值为-1表示写入失败
qint64 QIODevice::​write(const QByteArray & byteArray)

// 清空缓冲区
bool QFileDevice::​flush()

// 关闭数据流
void QIODevice::​close()[virtual]

dialog.cpp

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog)
{
    ui->setupUi(this);    connect(ui->pushButtonOpen,SIGNAL(clicked()),this,
            SLOT(btnClickedSlot()));
    connect(ui->pushButtonSave,SIGNAL(clicked()),this,
            SLOT(btnClickedSlot()));
    connect(ui->pushButtonCopy,SIGNAL(clicked()),this,
            SLOT(btnClickedSlot()));}Dialog::~Dialog()
{
    delete ui;
}void Dialog::printFileInfo()
{
    // 创建文件对象
    QFileInfo fileInfo(readPath);    // 判断文件是否存在
    if(!fileInfo.exists())
    {
        QMessageBox::warning(this,"提示","文件路径无效");
        return;
    }
    QString text = fileInfo.baseName();
    text.prepend("文件名称:");
    ui->textBrowserOpen->append(text);    qint64 size = fileInfo.size();
    text = QString::number(size);
    text.prepend("文件大小:").append("字节");
    ui->textBrowserOpen->append(text);    if(fileInfo.isReadable())
    {
        ui->textBrowserOpen->append("文件可读");
    }
    else
    {
        ui->textBrowserOpen->append("文件不可读");
    }
}void Dialog::copy()
{
    if(readPath == "")
    {
        QMessageBox::warning(this,"提示","请选择要读取的文件");
        return;
    }
    if(writePath == "")
    {
        QMessageBox::warning(this,"提示","请选择要写入的文件");
        return;
    }
    // 正在拷贝时屏蔽拷贝按钮
    ui->pushButtonCopy->setEnabled(false);    // 创建QFile对象
    QFile readFile(readPath);
    QFile writeFile(writePath);    // 打开文件流
    readFile.open(QIODevice::ReadOnly); // 只读模式
    writeFile.open(QIODevice::WriteOnly); // 只写模式    // 添加进度条进度
    qint64 totalSize = readFile.size(); // 获取文件总大小
    qint64 hasRead = 0; // 已经读写的大小    QByteArray array;
    while(!readFile.atEnd())
    {
        array = readFile.read(1024); // 每次读取1kb
        qint64 writeRet = writeFile.write(array); // 写入数据,返回值为本次写入的大小
        if(writeRet == -1) // 如果写入数据的返回值为-1表示写入失败
        {
            QMessageBox::critical(this,"错误","文件拷贝失败");
            return;
        }
        // 将写出的数据设置给进度条
        hasRead += writeRet; // 获取已经写入总大小
        int per = hasRead *100/totalSize; // 计算百分比
        ui->progressBar->setValue(per); // 设置给进度条
    }    // 清空缓存区
    writeFile.flush();    // 关闭数据流
    readFile.close();
    writeFile.close();    // 拷贝完成后解除按钮屏蔽
    ui->pushButtonCopy->setEnabled(true);
    // 添加拷贝完成提示框
    QMessageBox::information(this,"提示","拷贝完成");
}void Dialog::btnClickedSlot()
{
    if(ui->pushButtonOpen == sender())
    {
        QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";
        QString path = QFileDialog::getOpenFileName(this,"打开","D:/",filter);
        if(path != "")
        {
            readPath = path;
            ui->textBrowserOpen->append(path);
            printFileInfo(); // 文件读取成功后输出信息
        }
        else if(readPath == "")
        {
            QMessageBox::warning(this,"提示","请选择打开的文件");
        }    }
    else if(ui->pushButtonSave == sender())
    {
        QString filter = "所有文件(*.*);;Qt(*.cpp *.h *.pro *.ui)";
        QString path = QFileDialog::getSaveFileName(this,"保存","D:/",filter);
        if(path != "")
        {
            writePath = path;
            ui->textBrowserSave->append(path);
        }
        else if(writePath == "")
        {
            QMessageBox::warning(this,"提示","请选择保存的文件");
        }
    }
    else if(ui->pushButtonCopy == sender())
    {
        copy();
    }}

[思考]上面的代码有没有问题?

当拷贝大文件时,会出现卡顿,如果尝试关闭,则触发:

4、UI与耗时操作

在默认情况下,Qt的项目是单线程,这个自带的线程用于处理程序的主要任务和UI交互也被称为主线程或UI线程。

如果在主线程中执行耗时操作(IO或复杂算法)会导致主线程原本执行的操作被阻塞,甚至无法关闭,形成”假死“的现象。

当操作系统发现某个进程无法被正常关闭时,会弹出程序未响应窗口引导用户选择是否强制关闭当前进程。

解决方法是使用多线程。

5、QThread线程类

5.1 复现程序未响应

QThread类是Qt的线程类,

// 强制线程休眠msecs个毫秒
void	msleep(unsigned long msecs)

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog)
{
    ui->setupUi(this);
    connect(ui->pushButtonSleep,SIGNAL(clicked()),
            this,SLOT(btnSleepClickedSlot()));
    connect(ui->pushButtonClose,SIGNAL(clicked()),
            this,SLOT(close()));}Dialog::~Dialog()
{
    delete ui;
}void Dialog::btnSleepClickedSlot()
{
    qDebug() << "开始睡觉" ;
    QThread::msleep(15000);
    qDebug() << "睡眠结束" ;
}

5.2 创建并启动一个子线程

主线程以外的线程都是子线程,子线程不能执行主线程的UI操作。只能执行耗时操作。

下面是创建并启动一个自定义子线程的步骤:

  1. 在Qt Creator中选中项目名称,鼠标右键,点击添加新文件。

  1. 在弹出的窗口中,先设置类名,然后在选择基类名称QObject,最后点击”下一步“。

  1. 在项目管理界面,直接点击完成,可以看到线程类的文件已经创建。

  1. 选择新建的头文件,把继承的QObject更改为QThread

  1. 选择新建的.Cpp文件,把透传构造的QObject更改为QThread

  1. 在自定义线程中,覆盖基类QThread的run函数。

// 此函数是子线程执行的起始点,也是子线程的结束点。
void QThread::​run()[vritual protected]

7、在run函数的函数体中编写子线程要执行的耗时操作。

  1. 在主线程中创建子线程并启动

// 启动子线程,调用此函数后,会在子线程中自动执行run函数
// 参数欸子线程执行的优先级,默认值为创建所在的线程相同优先级
void QThread::​start(Priority priority = InheritPriority)[slot]

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog)
{
    ui->setupUi(this);
    connect(ui->pushButtonSleep,SIGNAL(clicked()),
            this,SLOT(btnSleepClickedSlot()));
    connect(ui->pushButtonClose,SIGNAL(clicked()),
            this,SLOT(close()));
}Dialog::~Dialog()
{
    delete ui;
}void Dialog::btnSleepClickedSlot()
{
     // 创建子线程对象
     MyThread *mt = new MyThread(this);
     // 启动子线程
     mt->start();
}

5.3 异步刷新

在实际的开发中,两个线程不可能毫无关系的前提下各干各的,最常见的情况是主线程分配一个耗时任务给子线程,子线程需要把耗时任务的执行情况反馈给主线程。主线程刷新子线程耗时的操作,并展示对应的UI效果。

【例子】:子线程执行文件拷贝,主线程显示拷贝的进度。

通常子线程是主线程对象的子对象,因此异步刷新就是对象通信的问题。使用信号槽解决。

咱们写一个简单的例子,一个伪拷贝的案例,使用for循环加睡眠, 模拟文件拷贝的功能。

今晚作业:真正的子线程拷贝文件。

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::Dialog)
{
    ui->setupUi(this);
    connect(ui->pushButton,SIGNAL(clicked()),
            this,SLOT(btnSleepClickedSlot()));
}Dialog::~Dialog()
{
    delete ui;
}void Dialog::btnSleepClickedSlot()
{
    ui->pushButton->setEnabled(false);
     // 创建子线程对象
     MyThread *mt = new MyThread(this);
     connect(mt,SIGNAL(valueSignal(int)),this,
             SLOT(valueSlot(int)));
     // 启动子线程
     mt->start();
}void Dialog::valueSlot(int value)
{
    ui->progressBar->setValue(value);
    if(value == 100)
    {
        ui->pushButton->setEnabled(true);
        QMessageBox::warning(this,"提示","拷贝完成");
    }
}

问题:频繁抖动窗口,会出现焦点抢夺问题,导致卡死。使用hide函数,来隐藏主窗口。

5.4 线程停止

子线程往往执行耗时操作,耗时操作又往往伴随着循环,因此并不建议使用粗暴的方式直接停止线程,因为强行停止线程会导致耗时操作资源无法回收等问题。

可以通过给循环设置标志位的方式使线程停止。

#include "dialog.h"
#include "ui_dialog.h"Dialog::Dialog(QWidget *parent) :QDialog(parent),ui(new Ui::Dialog)
{
    ui->setupUi(this);connect(ui->pushButton,SIGNAL(clicked()),this,SLOT(btnSleepClickedSlot()));
}Dialog::~Dialog()
{delete ui;
}void Dialog::btnSleepClickedSlot()
{if(ui->pushButton->text() == "开始拷贝"){// 创建子线程对象
        mt = new MyThread(this);connect(mt,SIGNAL(valueSignal(int)),this,SLOT(valueSlot(int)));// 启动子线程
        mt->start();
        ui->pushButton->setText("停止拷贝");}else if(ui->pushButton->text() == "停止拷贝"){
        ui->pushButton->setText("开始拷贝");
        mt->setRunningState(false);}}void Dialog::valueSlot(int value)
{
    ui->progressBar->setValue(value);if(value == 100){this->hide(); // 隐藏主窗口this->show(); // 显示主窗口
        QMessageBox::warning(this,"提示","拷贝完成");}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/702876.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java面试(网络)

TCP和UDP有什么区别&#xff1f;TCP三次握手不是两次&#xff1f; TCP&#xff1a;面向连接&#xff0c;可靠的&#xff0c;传输层通信协议。点对点&#xff0c;占用资源多&#xff0c;效率低。 UDP&#xff1a;无连接&#xff0c;不可靠&#xff0c;传输层通信协议。广播&…

考研408深度分析+全年规划

408确实很难&#xff0c;他的难分两方面 一方面是408本身的复习难度&#xff0c;我们都知道&#xff0c;408的考察科目有四科&#xff0c;分别是数据结构&#xff0c;计算机组成原理&#xff0c;操作系统和计算机网络。大家回想一下自己在大学本科时候学习这些专业课的难度&am…

Java:性能优化细节21-30

Java&#xff1a;性能优化细节21-30 21、ArrayList & LinkedList 一个是线性表&#xff0c;一个是链表&#xff0c;一句话&#xff0c;随机查询尽量使用ArrayList&#xff0c;ArrayList优于LinkedList&#xff0c;LinkedList还要移动指针&#xff0c;添加删除的操作Linke…

五、ChatGPT有哪些固有缺陷?—我耀学IT

上一讲&#xff0c;我们了解了ChatGPT工作的原理&#xff0c;这一讲&#xff0c;我们来看看它所存在的问题。 我们会看到&#xff0c;ChatGPT回答问题、写文章&#xff0c;有的时候很靠谱&#xff0c;有的时候又像在胡说八道&#xff0c;那么这些问题能不能解决呢&#xff1f; …

HTTPS对HTTP的加密过程

1、HTTPS是在HTTP的基础上&#xff0c;引入了一个加密层&#xff08;SSL&#xff09;&#xff0c;对数据进行保护&#xff0c;HTTP 是明文传输的&#xff08;不安全&#xff0c;很可能会被运营商通过referer劫持&#xff0c;或者黑客通过修改链接来窃数据&#xff09; 2、加密…

JavaSE——面向对象基础(4/4)-成员变量和局部变量的区别、面向对象综合案例(电影信息系统)

目录 补充&#xff1a;成员变量和局部变量的区别 面向对象综合案例 设计一个电影类 IDEA快捷操作 设计一个电影操作类 准备电影数据 业务处理 运行结果 补充&#xff1a;成员变量和局部变量的区别 区别成员变量&#xff08;对象的属性&#xff09;局部变量类中位置不同…

【数据结构】双向链表

一、main函数 #include <stdio.h> #include "./3.doublelinklist.h" int main(int argc, const char *argv[]) {doublelinklist* head creatr_doublelinklist();insertHead_doublelinklist(head,999);insertHead_doublelinklist(head,888);insertHead_double…

家装服务管理:Java技术的创新应用

✍✍计算机毕业编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java、…

SQL注入之oracle注入+SQLbypass+sqlmap实战

学习路还很长&#xff0c;切莫轻言放弃&#xff01; 目录 Oracle数据库介绍 Oracle数据库和MySQL数据库的差别 Oracle数据库注入 SQLbypass姿势 sqlmap工具实战(kali自带) Oracle数据库介绍 Oracle数据库是全球最知名的关系型数据库管理系统&#xff08;RDBMS&#xff09…

29-资源清单的管理工具-helm

一、helm的介绍 1&#xff0c;helm的价值概述 如下图所示&#xff0c;在一个企业中&#xff0c;可能存在多个不同的应用业务&#xff0c;每个业务可能包含多至十几、甚至几十个资源清单&#xff0c;那么对于“运维”和“研发”人员来讲&#xff0c;这么多的资源清单&#xff0…

说说UE5中的几种字符串类

在Unreal Engine 5 (UE5) 的C中&#xff0c;与字符串相关的类主要包括&#xff1a; FString&#xff1a; Unreal Engine中用于处理字符串的主要类&#xff0c;提供了丰富的字符串操作方法和功能。 FText&#xff1a; 用于表示本地化文本的类&#xff0c;可以包含多种语言的文本…

分布式知识整理

分布式锁 以商场系统超卖现象举例 超卖现象一 现象&#xff1a; 商品卖出数量超出了库存数量。 产生原因&#xff1a; 扣减库存的动作在程序中进行&#xff0c;在程序中计算剩余库存&#xff0c;在并发场景下&#xff0c;导致库存计算错误。 代码复现 es.shutdown(); cycl…

Nest.js权限管理系统开发(五)返回格式化

返回格式化拦截器 在上一篇《Nest.js权限管理系统开发&#xff08;四&#xff09;Swagger API接入》中&#xff0c;我们在base.controller.ts中创建了多个接口&#xff0c;每个接口都有不同的返回类型。现实中我们往往需要统一返回数据的格式&#xff0c;例如&#xff1a; {&…

【uni-app】路由

&#xff08;1&#xff09;路由配置 uni-app 页面路由全部交给框架统一管理&#xff0c;开发者需要在pages.json里配置每个路由页面的路径及页面样式&#xff08;类似小程序在 app.json 中配置页面路由&#xff09;。 "pages": [{"path": "pages/ind…

蓝桥杯倒计时47天!DFS基础——图的遍历

倒计时47天&#xff01; 深度优先搜索——DFS 温馨提示&#xff1a;学习dfs之前最好先了解一下递归的思想。 DFS基础——图的遍历 仙境诅咒 问题描述 在一片神秘的仙境中&#xff0c;有N位修仙者&#xff0c;他们各自在仙境中独立修炼&#xff0c;拥有自己独特的修炼之道…

websocket在django中的运用

14-2 聊天室实现思路&#xff1a;轮训、长轮训、websocket_哔哩哔哩_bilibili 参考大佬的B站学习笔记 https://www.cnblogs.com/wupeiqi/p/6558766.html 参考博客 https://www.cnblogs.com/wupeiqi/articles/9593858.html 参考博客 http: 是短连接&#xff0c;无状态的&…

探索网络通信的遗产:AppleTalk Data Stream Protocol (ADSP) 的全面解析

ADSP简介 AppleTalk Data Stream Protocol (ADSP) 是AppleTalk网络协议套件的一部分&#xff0c;设计用于在AppleTalk网络中提供端到端的可靠数据流服务。在1980年代和1990年代&#xff0c;AppleTalk是Apple计算机用于局域网通信的主要网络技术。ADSP提供了一种类似于现代TCP协…

conda 导出/导出配置好的虚拟环境

一. 导出环境配置&#xff08;yml文件&#xff09; 1. 在主目录下激活虚拟环境&#xff08;UE4是我的虚拟环境名称&#xff0c;请根据你自己的名称进行修改&#xff09; conda activate UE4 2. 运行此代码 conda env export > environment.yml 二. 导入环境配置&#xf…

创建第一个React项目

React脚手架 npx create-react-app react-demonpx是直接从互联网网上拉最新的脚手架进行创建react 运行React项目 npm start若想找到Webpack配置文件 npm ejectReact的基本使用 基本步骤 导入react和react-dom vue 创建react元素 渲染react元素到页面中导入 import React…

python统计分析——多解释变量的方差分析

参考资料&#xff1a;用python动手学统计学 1、导入库 # 导入库 # 用于数值计算的库 import numpy as np import pandas as pd import scipy as sp from scipy import stats # 用于绘图的库 from matplotlib import pyplot as plt import seaborn as sns sns.set() # 用于估计…