Evgenii Legotckoi
28 августа 2015 г. 19: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. Также объявляются слоты обработчиков нажатий кнопок.

  1. #ifndef MAINWINDOW_H
  2. #define MAINWINDOW_H
  3.  
  4. #include <QMainWindow>
  5. #include <QFileDialog>
  6. #include <QXmlStreamWriter>
  7. #include <QXmlStreamReader>
  8. #include <QXmlStreamAttribute>
  9. #include <QMessageBox>
  10. #include <QFile>
  11.  
  12. namespace Ui {
  13. class MainWindow;
  14. }
  15.  
  16. class MainWindow : public QMainWindow
  17. {
  18. Q_OBJECT
  19.  
  20. public:
  21. explicit MainWindow(QWidget *parent = 0);
  22. ~MainWindow();
  23.  
  24. private slots:
  25. /* Слоты обработчиков кнопок, работающих с записью в файл */
  26. void on_generateButton_clicked();
  27. void on_dialogButton_clicked();
  28.  
  29. /* Слоты обработчиков кнопок, работающих с чтением из файла */
  30. void on_readButton_clicked();
  31. void on_dialogReadButton_clicked();
  32.  
  33. private:
  34. Ui::MainWindow *ui;
  35. };
  36.  
  37. #endif // MAINWINDOW_H

mainwindow.cpp

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

  1. #include "mainwindow.h"
  2. #include "ui_mainwindow.h"
  3.  
  4. MainWindow::MainWindow(QWidget *parent) :
  5. QMainWindow(parent),
  6. ui(new Ui::MainWindow)
  7. {
  8. ui->setupUi(this);
  9. }
  10.  
  11. MainWindow::~MainWindow()
  12. {
  13. delete ui;
  14. }
  15.  
  16. /* Метод, выполняющий запись информации в XML-файл
  17. * */
  18. void MainWindow::on_generateButton_clicked()
  19. {
  20. /* Открываем файл для Записи с помощью пути, указанного в lineEditWrite */
  21. QFile file(ui->lineEditWrite->text());
  22. file.open(QIODevice::WriteOnly);
  23.  
  24. /* Создаем объект, с помощью которого осуществляется запись в файл */
  25. QXmlStreamWriter xmlWriter(&file);
  26. xmlWriter.setAutoFormatting(true); // Устанавливаем автоформатирование текста
  27. xmlWriter.writeStartDocument(); // Запускаем запись в документ
  28. xmlWriter.writeStartElement("resources"); // Записываем первый элемент с его именем
  29.  
  30. xmlWriter.writeStartElement("checkBox_1"); // Записываем тег с именем для первого чекбокса
  31. /* На основе состояния чекбокса записываем атрибут "boolean"
  32. * с указанием состояния чекбокса в этом атрибуте
  33. * */
  34. xmlWriter.writeAttribute("boolean",
  35. (ui->checkBox->isChecked() ? "true" : "false"));
  36. /* Записываем также в тело этого элемента строку из соответствующего lineEdit
  37. * */
  38. xmlWriter.writeCharacters(ui->lineEditCB1->text());
  39. xmlWriter.writeEndElement(); // Закрываем тег
  40.  
  41.  
  42. /* Повторяем те же действия для двух других чекбоксов
  43. * */
  44. xmlWriter.writeStartElement("checkBox_2");
  45. xmlWriter.writeAttribute("boolean",
  46. (ui->checkBox_2->isChecked() ? "true" : "false"));
  47. xmlWriter.writeCharacters(ui->lineEditCB2->text());
  48. xmlWriter.writeEndElement();
  49.  
  50. xmlWriter.writeStartElement("checkBox_3");
  51. xmlWriter.writeAttribute("boolean",
  52. (ui->checkBox_3->isChecked() ? "true" : "false"));
  53. xmlWriter.writeCharacters(ui->lineEditCB3->text());
  54. xmlWriter.writeEndElement();
  55.  
  56. /* Закрываем тег "resources"
  57. * */
  58. xmlWriter.writeEndElement();
  59. /* Завершаем запись в документ
  60. * */
  61. xmlWriter.writeEndDocument();
  62. file.close(); // Закрываем файл
  63. }
  64.  
  65. /* Метод, вызывающий диалоговое окно выбора файла для сохранения данных
  66. * */
  67. void MainWindow::on_dialogButton_clicked()
  68. {
  69. /* Вызываем диалог выбора файла для сохранения */
  70. QString filename = QFileDialog::getSaveFileName(this,
  71. tr("Save Xml"), ".",
  72. tr("Xml files (*.xml)"));
  73. /* Устанавливаем в lineEditWrite путь к файлу, с которым будем работать */
  74. if(filename != ""){
  75. ui->lineEditWrite->setText(filename);
  76. }
  77. }
  78.  
  79. void MainWindow::on_readButton_clicked()
  80. {
  81. /* Открываем файл для Чтения с помощью пути, указанного в lineEditWrite */
  82. QFile file(ui->lineEditRead->text());
  83. if (!file.open(QFile::ReadOnly | QFile::Text))
  84. {
  85. QMessageBox::warning(this,
  86. "Ошибка файла",
  87. "Не удалось открыть файл",
  88. QMessageBox::Ok);
  89. } else {
  90. /* Создаем объект, с помощью которого осуществляется чтение из файла */
  91. QXmlStreamReader xmlReader;
  92. xmlReader.setDevice(&file);
  93. xmlReader.readNext(); // Переходит к первому элементу в файле
  94.  
  95. /* Крутимся в цикле до тех пор, пока не достигнем конца документа
  96. * */
  97. while(!xmlReader.atEnd())
  98. {
  99. /* Проверяем, является ли элемент началом тега
  100. * */
  101. if(xmlReader.isStartElement())
  102. {
  103. /* Проверяем, относится ли тег к одному из чекбоксов.
  104. * Если "ДА", то выполняем проверку атрибута чекбокса
  105. * и записи для lineEdit
  106. * */
  107. if(xmlReader.name() == "checkBox_1")
  108. {
  109. /* Забираем все атрибуты тега и перебираем их для проверки на соответствие
  110. * нужному нам атрибуту
  111. * */
  112. foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
  113. /* Если найден нужный атрибут, то по его значению устанавливаем
  114. * состояние чекбокса
  115. * */
  116. if (attr.name().toString() == "boolean") {
  117. QString attribute_value = attr.value().toString();
  118. ui->checkBox->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
  119.  
  120. }
  121. }
  122. /* забираем текст из тела тега и вставляем его соответствующий lineEdit
  123. * */
  124. ui->lineEditCB1->setText(xmlReader.readElementText());
  125.  
  126. /* аналогично работаем с остальными тегами */
  127. } else if (xmlReader.name() == "checkBox_2"){
  128. foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
  129. if (attr.name().toString() == "boolean") {
  130. QString attribute_value = attr.value().toString();
  131. ui->checkBox_2->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
  132.  
  133. }
  134. }
  135. ui->lineEditCB2->setText(xmlReader.readElementText());
  136. } else if (xmlReader.name() == "checkBox_3"){
  137. foreach(const QXmlStreamAttribute &attr, xmlReader.attributes()) {
  138. if (attr.name().toString() == "boolean") {
  139. QString attribute_value = attr.value().toString();
  140. ui->checkBox_3->setChecked((QString::compare(attribute_value , "true") == 0) ? true : false);
  141.  
  142. }
  143. }
  144. ui->lineEditCB3->setText(xmlReader.readElementText());
  145. }
  146. }
  147. xmlReader.readNext(); // Переходим к следующему элементу файла
  148. }
  149. file.close(); // Закрываем файл
  150.  
  151. /* В данном коде не осуществляется проверка на закрытие тега
  152. * поскольку в этом нет необходимости, но функционал QXmlStreamReader это позволяет
  153. * */
  154. }
  155. }
  156.  
  157. /* Метод, вызывающий диалоговое окно выбора файла для чтения данных
  158. * */
  159. void MainWindow::on_dialogReadButton_clicked()
  160. {
  161. /* Вызываем диалог выбора файла для чтения */
  162. QString filename = QFileDialog::getOpenFileName(this,
  163. tr("Open Xml"), ".",
  164. tr("Xml files (*.xml)"));
  165. /* Устанавливаем в lineEditRead путь к файлу, с которым будем работать */
  166. if(filename != ""){
  167. ui->lineEditRead->setText(filename);
  168. }
  169. }

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

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

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <resources>
  3. <checkBox_1 boolean="false">XML</checkBox_1>
  4. <checkBox_2 boolean="false">Example</checkBox_2>
  5. <checkBox_3 boolean="true">EVILEG</checkBox_3>
  6. </resources>

Итог

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

Приложение для работы с XML файлами

Вам это нравится? Поделитесь в социальных сетях!

C
  • 21 января 2017 г. 0:40

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

Evgenii Legotckoi
  • 21 января 2017 г. 10:48

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

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

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

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
  • Последние комментарии
  • Evgenii Legotckoi
    16 апреля 2025 г. 17:08
    Благодарю за отзыв. И вам желаю всяческих успехов!
  • IscanderChe
    12 апреля 2025 г. 17:12
    Добрый день. Спасибо Вам за этот проект и отдельно за ответы на форуме, которые мне очень помогли в некоммерческих пет-проектах. Профессиональным программистом я так и не стал, но узнал мно…
  • AK
    1 апреля 2025 г. 11:41
    Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
  • Evgenii Legotckoi
    9 марта 2025 г. 21:02
    К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
  • VP
    9 марта 2025 г. 16:14
    Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…