Qt QFile文件操作的具體使用
很多應(yīng)用程序都需要具備操作文件的能力,包括對文件內(nèi)容進(jìn)行讀/寫、創(chuàng)建和刪除文件等,甚至某些應(yīng)用程序的誕生純粹是為了操作文件,比如 WPS Office、PDFedit 等。為此,Qt 框架提供了 QFile 類專門用來操作文件。
QFile文件操作
QFile 類支持對文件進(jìn)行讀取、寫入、刪除、重命名、拷貝等操作,它既可以操作文件文件,也可以操作二進(jìn)制文件。
使用 QFile 類操作文件之前,程序中需引入<QFile>頭文件。創(chuàng)建 QFile 類的對象,常用的構(gòu)造函數(shù)有:
QFile::QFile() QFile::QFile(const QString &name)
參數(shù) name 用來指定要操作的目標(biāo)文件,包含文件的存儲路徑和文件名,存儲路徑可以使用絕對路徑(比如 "D:/Demo/test.txt")或者相對路徑(比如"./Demo/test.txt"),路徑中的分隔符要用 "/" 表示。
通常情況下,我們會調(diào)用第二個(gè)構(gòu)造函數(shù),直接指明要操作的文件。對于第一個(gè)構(gòu)造函數(shù)創(chuàng)建的 QFile 對象,需要再調(diào)用 setFileName() 方法指明要操作的文件。
與 C++ 讀寫文件的規(guī)則一樣,使用 QFile 讀寫文件之前必須先打開文件,調(diào)用 open() 成員方法即可,常用的語法格式為:
bool QFile::open(OpenMode mode)
mode 參數(shù)用來指定文件的打開方式,下表羅列了此參數(shù)的可選值以及各自的含義:
打開方式 | 含 義 |
---|---|
QIODevice::ReadOnly | 只能對文件進(jìn)行讀操作 |
QIODevice::WriteOnly | 只能對文件進(jìn)行寫操作,如果目標(biāo)文件不存在,會自行創(chuàng)建一個(gè)新文件。 |
QIODevice::ReadWrite | 等價(jià)于 ReadOnly | WriteOnly,能對文件進(jìn)行讀和寫操作。 |
QIODevice::Append | 以追加模式打開文件,寫入的數(shù)據(jù)會追加到文件的末尾(文件原有的內(nèi)容保留)。 |
QIODevice::Truncate | 以重寫模式打開,寫入的數(shù)據(jù)會將原有數(shù)據(jù)全部清除。注意,此打開方式不能單獨(dú)使用,通常會和?ReadOnly 或?WriteOnly 搭配。 |
QIODevice::Text | 讀取文件時(shí),會將行尾結(jié)束符(Unix 系統(tǒng)中是 "\n",Windows 系統(tǒng)中是 "\r\n")轉(zhuǎn)換成‘\n';將數(shù)據(jù)寫入文件時(shí),會將行尾結(jié)束符轉(zhuǎn)換成本地格式,例如 Win32 平臺上是‘\r\n'。 |
表 1 QFile文件打開方式
根據(jù)需要,可以為 mode 參數(shù)一次性指定多個(gè)值,值和值之間用|分割。比如:
- QIODevice::ReadOnly | QIODevice::Text:表示只允許對文件進(jìn)行讀操作,讀取文件時(shí),會將行尾結(jié)束符轉(zhuǎn)換為 '\n';
- QIODevice::WriteOnly | QIODevice::Text:表示只允許對文件進(jìn)行寫操作,將數(shù)據(jù)寫入文件時(shí),會將行尾結(jié)束符轉(zhuǎn)換為本地格式;
- QIODevice::ReadWrite | QIODevice::Append | QIODevice::Text:表示對文件進(jìn)行寫操作,寫入的數(shù)據(jù)會存放到文件的尾部,同時(shí)數(shù)據(jù)中的行尾結(jié)束符轉(zhuǎn)換為本地格式。
注意,傳遞給 mode 參數(shù)的多個(gè)值之間不能相互沖突,比如 Append 和 Truncate 不能同時(shí)使用。
如果文件成功打開,open() 函數(shù)返回 true,否則返回 false。
QFile 類提供了很多功能實(shí)用的方法,可以快速完成對文件的操作,下表列舉了常用的一些:
普通成員方法 | 功 能 |
---|---|
qint64 QFile::size() const | 獲取當(dāng)前文件的大小。對于打開的文件,該方法返回文件中可以讀取的字節(jié)數(shù)。 |
bool QIODevice::getChar(char *c) | 從文件中讀取一個(gè)字符,并存儲到 c 中。讀取成功時(shí),方法返回 true,否則返回 false。 |
bool QIODevice::putChar(char c) | 向文件中寫入字符 c,成功時(shí)返回 true,否則返回 false。 |
QByteArray QIODevice::read(qint64 maxSize) | 從文件中一次性最多讀取 maxSize 個(gè)字節(jié),然后返回讀取到的字節(jié)。 |
qint64 QIODevice::read(char *data, qint64 maxSize) | 從文件中一次性對多讀取 maxSize 個(gè)字節(jié),讀取到的字節(jié)存儲到 data 指針指定的內(nèi)存控件中。該方法返回成功讀取到的字節(jié)數(shù)。 |
QByteArray QIODevice::readAll() | 讀取文件中所有的數(shù)據(jù)。 |
qint64 QIODevice::readLine(char *data, qint64 maxSize) | 每次從文件中讀取一行數(shù)據(jù)或者讀取最多?maxSize-1 個(gè)字節(jié),存儲到 data 中。該方法返回實(shí)際讀取到的字節(jié)數(shù)。 |
qint64 QIODevice::write(const char *data, qint64 maxSize) | 向 data 數(shù)據(jù)一次性最多寫入 maxSize 個(gè)字節(jié),該方法返回實(shí)際寫入的字節(jié)數(shù)。? |
qint64 QIODevice::write(const char *data) | 將 data 數(shù)據(jù)寫入文件,該方法返回實(shí)際寫入的字節(jié)數(shù)。 |
qint64 QIODevice::write(const QByteArray &byteArray) | 將 byteArray 數(shù)組中存儲的字節(jié)寫入文件,返回實(shí)際寫入的字節(jié)數(shù)。 |
bool QFile::copy(const QString &newName) | 將當(dāng)前文件的內(nèi)容拷貝到名為 newName 的文件中,如果成功,方法返回 true,否則返回 false。 copy 方法在執(zhí)行復(fù)制操作之前,會關(guān)閉源文件。 |
bool QFile::rename(const QString &newName) | 對當(dāng)前文件進(jìn)行重命名,新名稱為 newName,成功返回 true,失敗返回 false。 |
bool QFile::remove() | 刪除當(dāng)前文件,成功返回 true,失敗返回 false。 |
表 2 QFile常用方法
【實(shí)例一】演示了 QFile 類讀寫文本文件的過程。
#include <QFile> #include <QDebug> int main(int argc, char *argv[]) { //創(chuàng)建 QFile 對象,同時(shí)指定要操作的文件 QFile file("D:/demo.txt"); //對文件進(jìn)行寫操作 if(!file.open(QIODevice::WriteOnly|QIODevice::Text)){ qDebug()<<"文件打開失敗"; } //向文件中寫入兩行字符串 file.write("C語言中文網(wǎng)\n"); file.write("http://c.biancheng.net"); //關(guān)閉文件 file.close(); //重新打開文件,對文件進(jìn)行讀操作 if(!file.open(QIODevice::ReadOnly|QIODevice::Text)){ qDebug()<<"文件打開失敗"; } //每次都去文件中的一行,然后輸出讀取到的字符串 char * str = new char[100]; qint64 readNum = file.readLine(str,100); //當(dāng)讀取出現(xiàn)錯(cuò)誤(返回 -1)或者讀取到的字符數(shù)為 0 時(shí),結(jié)束讀取 while((readNum !=0) && (readNum != -1)){ qDebug() << str; readNum = file.readLine(str,100); } file.close(); return 0; }
執(zhí)行程序,"C語言中文網(wǎng)" 和 "http://c.biancheng.net" 先寫入 D 盤的 demo.txt 文件,然后再從文件中將它們讀取出來。
【實(shí)例二】演示 QFile 讀寫二進(jìn)制文件的過程。
#include <QFile> #include <QDebug> int main(int argc, char *argv[]) { //指定要寫入文件的數(shù)據(jù) qint32 nums[5]={1,2,3,4,5}; //寫入文件之前,要將數(shù)據(jù)以二進(jìn)制方式存儲到字節(jié)數(shù)組中 QByteArray byteArr; byteArr.resize(sizeof(nums)); for(int i=0;i<5;i++){ //借助指針,將每個(gè)整數(shù)拷貝到字節(jié)數(shù)組中 memcpy(byteArr.data()+i*sizeof(qint32),&(nums[i]),sizeof(qint32)); } //將 byteArr 字節(jié)數(shù)組存儲到文件中 QFile file("D:/demo.dat"); file.open(QIODevice::WriteOnly); file.write(byteArr); file.close(); //再次打開文件,讀取文件中存儲的二進(jìn)制數(shù)據(jù) file.open(QIODevice::ReadOnly); QByteArray resArr = file.readAll(); //輸出讀取到的二進(jìn)制數(shù)據(jù) qDebug()<<"resArr: "<<resArr; //將二進(jìn)制數(shù)據(jù)轉(zhuǎn)化為整數(shù) char* data = resArr.data(); while(*data){ qDebug() << *(qint32*)data; data += sizeof(qint32); } return 0; }
執(zhí)行程序,demo.dat 文件中會存儲 {1,2,3,4,5} 這 5 個(gè)整數(shù)的二進(jìn)制形式,同時(shí)輸出以下內(nèi)容:
resArr:? "\x01\x00\x00\x00\x02\x00\x00\x00\x03\x00\x00\x00\x04\x00\x00\x00\x05\x00\x00\x00"
1
2
3
4
5
單獨(dú)使用 QFile 類讀寫文件的過程既繁瑣又復(fù)雜,Qt 提供了兩個(gè)輔助類 QTextStream 和 QDataStream,前者用來讀寫文件文件,后者用來讀寫二進(jìn)制文件,QFile 可以和它們搭配使用,從整體上提高讀寫文件的開發(fā)效率。
QFile+QTextStream
和單獨(dú)使用 QFile 類讀寫文本文件相比,QTextStream 類提供了很多讀寫文件相關(guān)的方法,還可以設(shè)定寫入到文件中的數(shù)據(jù)格式,比如對齊方式、寫入數(shù)字是否帶前綴等等。
使用 QTextStream 類之前,程序中要先引入<QTextStream>頭文件。QTextStream 類提供了很多種構(gòu)造函數(shù),常用的是:
QTextStream(QIODevice *device)
QIODevice 是 QFile 的父類,因此在構(gòu)造 QTextStream 類的對象時(shí),需要傳遞一個(gè) QFile 類的對象。
下表羅列了 QTextStream 類常用的一些方法:
成員方法 | 功 能 |
---|---|
bool QTextStream::atEnd() const | 判斷是否讀到文件末尾,如果已經(jīng)達(dá)到末尾,返回 true,否則返回 false。 |
QString QTextStream::read(qint64 maxlen) | 從文件中讀最多 maxlen 個(gè)字符,返回這些字符組成的 QString 字符串。 |
QString QTextStream::readAll() | 從文件中讀取所有內(nèi)容,返回由讀取內(nèi)容組成的 QString 字符串。 |
QString QTextStream::readLine(qint64 maxlen = 0) | 默認(rèn)讀取一行文本,如果手動指定 maxlen 的值,則最多讀取 maxlen 個(gè)字符,并返回讀取內(nèi)容組成的 QString 字符串。 |
void QTextStream::setFieldAlignment(FieldAlignment mode) | 設(shè)置對齊方式,通常與 setFieldWidth() 一起使用。 |
void QTextStream::setFieldWidth(int width) | 設(shè)置每份數(shù)據(jù)占用的位置寬度為 width。 |
表 3 QTextStream常用方法
QTextStream 類重載了>>輸入運(yùn)算符和>>輸出運(yùn)算符,使讀寫文本文件變得更簡單。例如,用 QTextStream 實(shí)現(xiàn)【實(shí)例一】的程序如下:
#include <QFile> #include <QDebug> #include <QString> #include <QTextStream> int main(int argc, char *argv[]) { //創(chuàng)建 QFile 對象,同時(shí)指定要操作的文件 QFile file("D:/demo.txt"); //對文件進(jìn)行寫操作 if(!file.open(QIODevice::WriteOnly|QIODevice::Text)){ qDebug()<<"文件打開失敗"; } QTextStream out(&file); //向文件中寫入兩行字符串 out << (QString)"C語言中文網(wǎng)\n" << (QString)"http://c.biancheng.net"; //關(guān)閉文件 file.close(); //重新打開文件,對文件進(jìn)行讀操作 if(!file.open(QIODevice::ReadOnly|QIODevice::Text)){ qDebug()<<"文件打開失敗"; } QTextStream in(&file); //一直讀,直至讀取失敗 while(!in.atEnd()){ QString str; //從文件中讀取一個(gè)字符串 in >> str; qDebug() << str; } file.close(); return 0; }
和<iostream>類似,QTextStream 類提供了兩種格式化輸出的方法,一種是調(diào)用該類的成員方法,例如表 3 中的 setFieldAlignment()、setFieldWidth 等,另一種是調(diào)用 QTextStream 類提供的格式描述符,下表羅列了常用的一些:
描述符 | 功能相同的方法 | 功 能 |
---|---|---|
Qt::hex | QTextStream::setIntegerBase(16) | 將指定整數(shù)對應(yīng)的 16 進(jìn)制數(shù)寫入到文件中。 |
Qt::showbase | QTextStream::setNumberFlags(numberFlags() | ShowBase) | 對于非十進(jìn)制數(shù),寫入到文件中時(shí)帶上相應(yīng)的前綴。二進(jìn)制數(shù)前綴是 0b,八進(jìn)制數(shù)前綴是 0,十六進(jìn)制數(shù)前綴是 0x。 |
Qt::forcesign | QTextStream::setNumberFlags(numberFlags() | ForceSign) | 將數(shù)字寫入文件時(shí),帶上正負(fù)號。 |
Qt::fixed | QTextStream::setRealNumberNotation(FixedNotation) | 將浮點(diǎn)數(shù)以普通小數(shù)的形式寫入文件。 |
Qt::scientific | QTextStream::setRealNumberNotation(ScientificNotation) | 將浮點(diǎn)數(shù)以科學(xué)計(jì)數(shù)法的形式寫入文件。 |
Qt::left | QTextStream::setFieldAlignment(AlignLeft) | 左對齊 |
Qt::right | QTextStream::setFieldAlignment(AlignRight) | 右對齊 |
Qt::center | QTextStream::setFieldAlignment(AlignCenter) | 居中對齊 |
表 4 QTextStream常用格式描述符
舉個(gè)簡單的例子:
#include <QFile> #include <QDebug> #include <QString> #include <QTextStream> int main(int argc, char *argv[]) { QFile file("D:/demo.txt"); if(!file.open(QIODevice::WriteOnly|QIODevice::Text)){ qDebug()<<"文件打開失敗"; } QTextStream out(&file); //將 10 的十六進(jìn)制數(shù)寫入文件 out << hex << 10; //設(shè)置每份數(shù)據(jù)占用 10 個(gè)字符的位置 out.setFieldWidth(10); //以右對齊的方式寫入 3.14 out << left << 3.14; //后續(xù)數(shù)據(jù)以左對齊的方式寫入文件 out.setFieldAlignment(QTextStream::AlignRight); out << 2.7; //關(guān)閉文件 file.close(); return 0; }
程序運(yùn)行后,demo.txt 存儲的文本內(nèi)容為:
a3.14? ? ? ? ? ? ?2.7
QFile+QDataStream
QDataStream 類的用法和 QTextStream 非常類似,最主要的區(qū)別在于,QDataStream 用于讀寫二進(jìn)制文件。
使用 QDataStream 類之前,程序中要引入<QDataStream>頭文件。創(chuàng)建 QDataStream 對象常用的構(gòu)造函數(shù)為:
QDataStream::QDataStream(QIODevice *d)
下表羅列了 QDataStream 類常用的成員方法:
成員方法 | 功 能 |
---|---|
bool QDataStream::atEnd() const | 判斷是否讀到文件末尾,如果已經(jīng)達(dá)到末尾,返回 true,否則返回 false。 |
QDataStream &QDataStream::readBytes(char *&s, uint &l) | 對于用 writeBytes() 方法寫入文件的?l 和?s,只能使用 readBytes() 方法讀取出來。? |
int QDataStream::readRawData(char *s, int len) | 從文件中讀取最多 len 字節(jié)的數(shù)據(jù)到 s 中,返回值表示實(shí)際讀取的字節(jié)數(shù)。注意,調(diào)用該方法之前,需要先給 s 參數(shù)分配好內(nèi)存空間。 |
void QDataStream::setVersion(int v) | 不同版本的 Qt 中,同名稱的數(shù)據(jù)類型也可能存在差異,通過調(diào)用此方法手動指定版本號,可以確保讀取數(shù)據(jù)的一致性。 |
int QDataStream::skipRawData(int len) | 跳過文件中的?len 個(gè)字節(jié),返回實(shí)際跳過的字節(jié)數(shù)。 |
QDataStream &QDataStream::writeBytes(const char *s, uint len) | 將長度 len 和 s 一起寫入到文件中,對于 writeBytes() 寫入的數(shù)據(jù),只能用 readBytes() 方法讀取。 |
int QDataStream::writeRawData(const char *s, int len) | 將 s 中前 len 字節(jié)的數(shù)據(jù)寫入文件,返回值表示成功寫入的字節(jié)數(shù)。 |
表 5 QDataStream常用方法
QDataStream 類也對<<和>>進(jìn)行了重載,舉個(gè)簡單的例子,用 QDataStream 重新實(shí)現(xiàn)實(shí)例二:
#include <QFile> #include <QDebug> #include <QDataStream> int main(int argc, char *argv[]) { //指定要寫入文件的數(shù)據(jù) qint32 nums[5]={1,2,3,4,5}; QFile file("D:/demo.dat"); file.open(QIODevice::WriteOnly); //創(chuàng)建 QDataStream 對象 QDataStream out(&file); //將 nums 數(shù)組中的整數(shù)逐個(gè)寫入到二進(jìn)制文件中 for(int i=0;i<5;i++){ out << nums[i]; } file.close(); //再次打開文件,讀取文件中存儲的二進(jìn)制數(shù)據(jù) file.open(QIODevice::ReadOnly); QDataStream in(&file); //讀取二進(jìn)制文件中的數(shù)據(jù) while(!in.atEnd()){ //每次讀取一個(gè)整數(shù) qint32 num; in >> num; qDebug() << num; } return 0; }
輸出結(jié)果為:
1
2
3
4
5
到此這篇關(guān)于Qt QFile文件操作的具體使用的文章就介紹到這了,更多相關(guān)Qt QFile文件操作內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
全面了解結(jié)構(gòu)體、聯(lián)合體和枚舉類型
下面小編就為大家?guī)硪黄媪私饨Y(jié)構(gòu)體、聯(lián)合體和枚舉類型。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-07-07C++使用MySQL-Connector/C++連接MySQL出現(xiàn)LNK2019錯(cuò)誤的解決方法
這篇文章主要介紹了C++使用MySQL-Connector/C++連接MySQL出現(xiàn)LNK2019錯(cuò)誤的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03C語言求Fibonacci斐波那契數(shù)列通項(xiàng)問題的解法總結(jié)
斐波那契數(shù)列相關(guān)問題是考研和ACM中常見的算法題目,這里特地為大家整理了C語言求Fibonacci斐波那契數(shù)列通項(xiàng)問題的解法總結(jié),需要的朋友可以參考下2016-06-06c++內(nèi)聯(lián)函數(shù)(inline)使用詳解
這篇文章主要介紹了c++內(nèi)聯(lián)函數(shù)(inline)使用詳解,需要的朋友可以參考下2014-04-04C語言輪轉(zhuǎn)數(shù)組的三種實(shí)現(xiàn)
輪轉(zhuǎn)數(shù)組是一種將數(shù)組元素循環(huán)移動的處理方式,它通常用于解決一些需要對固定長度的數(shù)組進(jìn)行循環(huán)滾動處理的問題,本文就介紹了C語言輪轉(zhuǎn)數(shù)組的三種實(shí)現(xiàn),感興趣的可以了解一下2023-08-08C語言實(shí)現(xiàn)餐飲結(jié)賬管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)餐飲結(jié)賬管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11