QTextStream записывает неверные данные в файл

Я пытаюсь читать и писать из одного и того же файла, но получаю странное поведение. Вот пример кода:

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QTextStream>
#include <QFile>

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
Q_OBJECT

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

private slots:
void on_textEdit_textChanged();

private:
QFile *File;
QTextStream *FileStream;
QString TextFromFile;
Ui::MainWindow *ui;
};

#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"#include "ui_mainwindow.h"
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);

File = new QFile;
File->setFileName("test.txt");
File->open(QIODevice::ReadWrite | QIODevice::Text);
FileStream = new QTextStream(File);

TextFromFile = FileStream->readAll();
ui->textEdit->setText(TextFromFile);

File->close();
File->open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text);
}

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

void MainWindow::on_textEdit_textChanged()
{
*FileStream << ui->textEdit->toPlainText();
FileStream->flush();
}

Поэтому, когда я набираю, например, это:
введите описание изображения здесь
файл изменится на это:
введите описание изображения здесь
но мне нужно это:
введите описание изображения здесь
Моя цель — перезаписывать файл каждый раз, когда я печатаю символы в textedit.

0

Решение

Если вам нужно каждый раз переписывать файл, попробуйте сделать это без указателей. Что-то вроде:

void MainWindow::on_textEdit_textChanged()
{
QElapsedTimer timer;
timer.start();
QFile file("test.txt");
if(!file.open(QIODevice::WriteOnly | QIODevice::Text))
qDebug() << "failed open";
else
{
QTextStream out(&file);
out << ui->textEdit->toPlainText();
}
qDebug() << "The slow operation took" << timer.elapsed() << "milliseconds";
}

QFile деструктор закроет файл в конце слота.

Вывод в моем случае:

The slow operation took 0 milliseconds
The slow operation took 0 milliseconds

Конечно, с большими данными это будет медленнее, но я думаю, что это нормальный подход.

Возможно, вы думали, что QIODevice::Truncate должен сделать свое дело, но это неправильно. Из документа:

Если это возможно, устройство усекается до его открытия. Все раньше
содержимое устройства теряется.

Но в вашем коде это не работает, потому что вы используете тот же поток, вы не открываете свой файл каждый раз, вы просто добавляете новые слова каждый раз. А также flush() просто сбрасывает любые буферизованные данные, ожидающие записи на устройство.

3

Другие решения


По вопросам рекламы [email protected]