Евгений Легоцкой28 серпня 2015 р. 08:34

Qt/C++ - Урок 011. XML файлы в Qt - чтение и запись

Средства разработки Qt включают в себя функционал по работе с XML файлами, который позволяет как создавать XML файлы, так и читать их, что может быть полезно для создания программ по автоматическому созданию XML разметок, конфигурационных файлов, использующих XML , либо для парсинга XML-файлов .

Для записи файлов в формате XML используется класс QXmlStreamWriter, тогда как для чтения XML файлов используется другой класс QXmlStreamReader.

Программный код был написан в QtCreator 3.3.1 на основе Qt 5.4.1.

Структура проекта для работы с XML файлом

В данном проекте используется минимум файлов:

  • XMLExample.pro - профайл;
  • mainwindow.h - заголовочный файл основного окна приложения;
  • mainwindow.cpp - исходный код окна;
  • main.cpp - основной исходный файл, с которого стартует приложение;
  • mainwindow.ui - формочка основного окна приложения;

mainwindow.ui

На этот раз в формочку главного окна необходимо добавить достаточно большое количество элементов, чтобы оно стало выглядеть следующим образом:

Главное окно приложения для работы с XML файлом

Список всех элементов, с которыми будет производиться работа в программном коде:

  • checkBox - первый чекбокс;
  • checkBox_1 - второй чекбокс;
  • checkBox_2 - третий чекбокс;
  • lineEditCB1 - lineEdit для первого чекбокса;
  • lineEditCB2 - lineEdit для второго чекбокса;
  • lineEditCB3 - lineEdit для третьего чекбокса;
  • lineEditRead - lineEdit, в котором указывается путь к файлу для чтения;
  • lineEditWrite - lineEdit, в котором указывается путь к файлу для записи;
  • dialogReadButton - кнопка для вызова диалога сохранения файла, в данном случае выполняется просто выбор файла, с которым будет производиться работа;
  • dialogButton - кнопка для вызова диалога записи файла, в данном случае выполняется просто выбор файла, с которым также будет производиться работа;
  • readButton - кнопка выполнения чтения из файла;
  • generateButton - кнопка для выполнения записи в файл.

mainwindow.h

В заголовочном файле подключаются библиотеки для работы с QXmlStreamWriter и QXmlStreamReader. Также объявляются слоты обработчиков нажатий кнопок.

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QFileDialog>
#include <QXmlStreamWriter>
#include <QXmlStreamReader>
#include <QXmlStreamAttribute>
#include <QMessageBox>
#include <QFile>

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

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

private slots:
    /* Слоты обработчиков кнопок, работающих с записью в файл */
    void on_generateButton_clicked();
    void on_dialogButton_clicked();

    /* Слоты обработчиков кнопок, работающих с чтением из файла */
    void on_readButton_clicked();
    void on_dialogReadButton_clicked();

private:
    Ui::MainWindow *ui;
};

#endif // MAINWINDOW_H

mainwindow.cpp

В данном исходном файле осуществляется чтение и запись данных в XML-файл, а также подстановка параметров из XML-файла в форм главного окна.

#include "mainwindow.h"
#include "ui_mainwindow.h"
    
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
}
    
MainWindow::~MainWindow()
{
    delete ui;
}
   
/* Метод, выполняющий запись информации в XML-файл
 * */
void MainWindow::on_generateButton_clicked()
{
    /* Открываем файл для Записи с помощью пути, указанного в lineEditWrite */
    QFile file(ui->lineEditWrite->text());
    file.open(QIODevice::WriteOnly);

    /* Создаем объект, с помощью которого осуществляется запись в файл */
    QXmlStreamWriter xmlWriter(&file);
    xmlWriter.setAutoFormatting(true);  // Устанавливаем автоформатирование текста
    xmlWriter.writeStartDocument();     // Запускаем запись в документ
    xmlWriter.writeStartElement("resources");   // Записываем первый элемент с его именем

    xmlWriter.writeStartElement("checkBox_1");  // Записываем тег с именем для первого чекбокса
    /* На основе состояния чекбокса записываем атрибут "boolean"
     * с указанием состояния чекбокса в этом атрибуте
     * */
    xmlWriter.writeAttribute("boolean",
                             (ui->checkBox->isChecked() ? "true" : "false"));
    /* Записываем также в тело этого элемента строку из соответствующего lineEdit
     * */
    xmlWriter.writeCharacters(ui->lineEditCB1->text());
    xmlWriter.writeEndElement();        // Закрываем тег


    /* Повторяем те же действия для двух других чекбоксов
     * */
    xmlWriter.writeStartElement("checkBox_2");
    xmlWriter.writeAttribute("boolean",
                             (ui->checkBox_2->isChecked() ? "true" : "false"));
    xmlWriter.writeCharacters(ui->lineEditCB2->text());
    xmlWriter.writeEndElement();

    xmlWriter.writeStartElement("checkBox_3");
    xmlWriter.writeAttribute("boolean",
                             (ui->checkBox_3->isChecked() ? "true" : "false"));
    xmlWriter.writeCharacters(ui->lineEditCB3->text());
    xmlWriter.writeEndElement();

    /* Закрываем тег "resources"
     * */
    xmlWriter.writeEndElement();
    /* Завершаем запись в документ
     * */
    xmlWriter.writeEndDocument();
    file.close();   // Закрываем файл
}
    
/* Метод, вызывающий диалоговое окно выбора файла для сохранения данных
 * */
void MainWindow::on_dialogButton_clicked()
{
    /* Вызываем диалог выбора файла для сохранения */
    QString filename = QFileDialog::getSaveFileName(this,
                                           tr("Save Xml"), ".",
                                           tr("Xml files (*.xml)"));
    /* Устанавливаем в lineEditWrite путь к файлу, с которым будем работать */
    if(filename != ""){
        ui->lineEditWrite->setText(filename);
    }
}
    
void MainWindow::on_readButton_clicked()
{
    /* Открываем файл для Чтения с помощью пути, указанного в lineEditWrite */
    QFile file(ui->lineEditRead->text());
    if (!file.open(QFile::ReadOnly | QFile::Text))
    {
        QMessageBox::warning(this,
                             "Ошибка файла",
                             "Не удалось открыть файл",
                             QMessageBox::Ok);
    } else {
        /* Создаем объект, с помощью которого осуществляется чтение из файла */
        QXmlStreamReader xmlReader;
        xmlReader.setDevice(&file);
        xmlReader.readNext();   // Переходит к первому элементу в файле

        /* Крутимся в цикле до тех пор, пока не достигнем конца документа
         * */
        while(!xmlReader.atEnd())
        {
            /* Проверяем, является ли элемент началом тега
             * */
            if(xmlReader.isStartElement())
            {
                /* Проверяем, относится ли тег к одному из чекбоксов.
                 * Если "ДА", то выполняем проверку атрибута чекбокса
                 * и записи для lineEdit
                 * */
                if(xmlReader.name() == "checkBox_1")
                {
                    /* Забираем все атрибуты тега и перебираем их для проверки на соответствие
                     * нужному нам атрибуту
                     * */
                    foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
                        /* Если найден нужный атрибут, то по его значению устанавливаем
                         * состояние чекбокса
                         * */
                        if (attr.name().toString() == "boolean") {
                            QString attribute_value = attr.value().toString();
                            ui->checkBox->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
    
                        }
                    }
                    /* забираем текст из тела тега и вставляем его соответствующий lineEdit
                     * */
                    ui->lineEditCB1->setText(xmlReader.readElementText());

                    /* аналогично работаем с остальными тегами */
                } else if (xmlReader.name() == "checkBox_2"){
                    foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
                        if (attr.name().toString() == "boolean") {
                            QString attribute_value = attr.value().toString();
                            ui->checkBox_2->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
    
                        }
                    }
                    ui->lineEditCB2->setText(xmlReader.readElementText());
                } else if (xmlReader.name() == "checkBox_3"){
                    foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
                        if (attr.name().toString() == "boolean") {
                            QString attribute_value = attr.value().toString();
                            ui->checkBox_3->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
    
                        }
                    }
                    ui->lineEditCB3->setText(xmlReader.readElementText());
                }
            }
            xmlReader.readNext(); // Переходим к следующему элементу файла
        }
        file.close(); // Закрываем файл
 
        /* В данном коде не осуществляется проверка на закрытие тега
         * поскольку в этом нет необходимости, но функционал QXmlStreamReader это позволяет
         * */
    }
}
    
/* Метод, вызывающий диалоговое окно выбора файла для чтения данных
 * */
void MainWindow::on_dialogReadButton_clicked()
{
    /* Вызываем диалог выбора файла для чтения */
    QString filename = QFileDialog::getOpenFileName(this,
                                       tr("Open Xml"), ".",
                                       tr("Xml files (*.xml)"));
   /* Устанавливаем в lineEditRead путь к файлу, с которым будем работать */
    if(filename != ""){
        ui->lineEditRead->setText(filename);
    }
}

В результате мы получаем возможность создать XML-файл с параметрами чекбоксов и наших объектов типа lineEdit , а также распарсить этот файл и установить полученные данные в соответствующие чекбоксы и объекты lineEdit.

Полученный файл должен будет выглядеть следующим образом:

<?xml version="1.0" encoding="UTF-8"?>
<resources>
    <checkBox_1 boolean="false">XML</checkBox_1>
    <checkBox_2 boolean="false">Example</checkBox_2>
    <checkBox_3 boolean="true">EVILEG</checkBox_3>
</resources>

Итог

В результате должно получиться приложение, которое создаёт или перезаписывает XML-файл , забирая данные из чекбоксов и соответствующих полей lineEdit , а также производящее парсинг этого XML-файла с установкой данных в чекбоксы и поля lineEdit .

Приложение для работы с XML файлами
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.
Підтримайте автора Donate
C

Не могли бы предоставить пример, где больше уровней в xml файле?

Это конечно можно, но насколько больше? И применительно к какому виду... Всё упирается же в то, куда именно будет этот XML применяться.

Например, чтение данных из XML можно сделать и с помощью QDomDocument. А если учесть, что SVG файл также фактически является XML документом, то Вы можете посмотреть следующий пример, для ознакомления с вытягиванием данных из XML файла:

SvgReader на Qt. Восстановление данных из файла SVG в QGraphicsScene

Коментарі

Only authorized users can post comments.
Please, Log in or Sign up
Timeweb

Дозвольте порекомендувати вам чудовий хостинг, на якому розташований EVILEG.

Протягом багатьох років Timeweb доводив свою стабільність.

Для проектів на Django рекомендую VDS хостинг

Переглянути хостинг
СП

C++ - Тест 001. Первая программа и типы данных

  • Результат:93бали,
  • Рейтинг балів8
VS

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:30бали,
  • Рейтинг балів-10
J

C++ - Тест 001. Первая программа и типы данных

  • Результат:93бали,
  • Рейтинг балів8
Останні коментарі

Qt/C++ - Урок 074. Генерація псевдовипадкових чисел, використання STD бібліотеки random

А использование функции global() не решает ли эти проблемы? value = QRandomGenerator::global()->bounded(15, 43); Получаемая последовательность каждый раз новая.

Qt/C++ - Урок 074. Генерація псевдовипадкових чисел, використання STD бібліотеки random

А использование функции global() не решает ли эти проблемы? value = QRandomGenerator::global()->bounded(15, 43); Получаемая последовательность каждый раз новая.
S

QML - Урок 026. Intents с Qt для Android, часть 1

Есть ли возможность приведения java типа у QAndroidJniObject? Интересует конкретно class to
ВК

Qt / C ++ - Урок 015. QTableWidget або Як зробити таблицю з чекбоксами

Кто-нибудь знает, как сделать так, чтобы в QTableWidget состоящей из чекбоксов в строке таблицы можно было выбрать только один checkbox ?

Qt/C++ - Урок 006. QSqlQueryModel - Таблицы в Qt с помощью SQL-запросов

QSqlTableModel выполняет ряд стандартных операций для одной таблицы из базы данных. Поэтому там и реализован функционал по удалению и редактированию. QSqlQueryModel позволяет выполнить запр…
Тепер обговоріть на форумі

Лишние строки при выборке из базы данных SQLite

без знаниния теории, или хотяб чаще заглядывать в документацию - практика большого опыта не даст. толк от написанного когда когда вы не понимаете что там написано и как работает? или не использо…
u
  • ubomj
  • 30 жовтня 2020 р. 07:56

Не проверять форму если нажали кнопку

что то не выходит(( попробовал вот так: def clean_title(self): title = self.cleaned_data.get('title') if 'title' in self.data: raise ValidationError("Titl…

QSqlTableModel не удаётся редактировать и удалять данные

а проверить просто что находится в базе? и вообще SQLite, на сколько помню, удаляет данные из поля, чтоб изменить размер и удалить пустые строки нужно доболнительно делать вакуум
D

LibreOffice QT Widget

Я бы хотел интегрировать приложения из LibreOffice в свою программу. В идеале использовать их как виджеты Наткнулся на пакет libreoffice-qt5, который вроде как позволяет это делать htt…

Создание черновика как на авито и тд

А черновик в свою очередь нужен пока только для получения id, который нужен для мультизагружки изображений и привязки их к посту. как то так... я бы вообще решал это так: class P…
Про
Послуги
© EVILEG 2015-2020
Рекомендуємо хостинг TIMEWEB