欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++讀入XML文件示例

 更新時(shí)間:2016年12月05日 08:22:22   作者:新科程序猿  
本篇文章主要介紹了C++讀入XML文件,讀取和設(shè)置xml配置文件是最常用的操作,TinyXML是一個(gè)開(kāi)源的解析XML的C++解析庫(kù),感興趣的小伙伴們可以參考一下。

最近要做一個(gè)VRP的算法,測(cè)試集都是放在Xml文件中,而我的算法使用C++來(lái)寫,所以需要用C++來(lái)讀取Xml文件。

在百度上搜“C++讀取Xml文件”,可以出來(lái)很多博客,大多數(shù)是關(guān)于tinyXml的,所以這篇博文也是講述如何用tinyXML來(lái)讀取XML文件。

tinyXml是一個(gè)免費(fèi)開(kāi)源的C++庫(kù),可以到官網(wǎng)上下載:https://sourceforge.net/projects/tinyxml/。

下載下來(lái)解壓之后,可以看到下面這些文件:

我是在windows下用VS來(lái)寫C++的,按照@marchtea的說(shuō)法,只需要直接打開(kāi)tinyxml.sln就可以,不過(guò)我還是用了笨辦法:

  • 把tinystr.cpp, tinyxml.cpp, tinyxmlerror.cpp, tinyxmlparser.cpp, tinystr.h, tinyxml.h拷貝到工程目錄下;
  • 然后加入頭文件引用:#include "tinystr.h"   #include "tinyxml.h"。

接下來(lái)就來(lái)分享一下我讀取VRP問(wèn)題中的solomon benchmark的方法,這些方法都是參考自tinyXml的官方教程,在下載的文件夾中有"doc"子文件夾,打開(kāi)它,有一個(gè)叫做"tutorial0"的html文件,打開(kāi)它可以看到詳細(xì)的教程。

OK,now begins!

我要讀取的Xml文件有如下的格式(只列舉部分):

<!-- 要讀取的Xml文件 -->

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>

<instance>

  <network>

    <nodes>

      <node id="0" type="0">

        <cx>40.0</cx>

        <cy>50.0</cy>

      </node>

    <!-- 有N+1個(gè)這樣的node節(jié)點(diǎn) -->

    </nodes>  

  </network>

  <requests>

    <request id="1" node="1">

      <tw>

        <start>145</start>

        <end>175</end>

      </tw>

      <quantity>20.0</quantity>

      <service_time>10.0</service_time>

    </request>

    <!-- 有N個(gè)這樣的request節(jié)點(diǎn) -->

  </requests>

</instance>

這里稍微解釋一下為什么nodes節(jié)點(diǎn)的數(shù)目會(huì)比requests節(jié)點(diǎn)多1個(gè)。這是因?yàn)閚odes節(jié)點(diǎn)包括了顧客節(jié)點(diǎn)(N個(gè))和倉(cāng)庫(kù)節(jié)點(diǎn)(1個(gè)),而requests屬性只屬于顧客節(jié)點(diǎn)。

我是把xml文件中的這些數(shù)據(jù)讀入到類對(duì)象數(shù)組中,每個(gè)類對(duì)象代表一個(gè)節(jié)點(diǎn),類的定義如下:

// Customer.h

#ifndef _Customer_H

#define _Customer_H

 

class Customer{

public:

  Customer(int id=0, float x=0, float y=0, float startTime=0, float endTime=0, float quantity=0, float serviceTime=0);

  void setId(int id);  // 設(shè)置成員id的值

  void setX(float x);  // 設(shè)置成員x的值

  void setY(float y);  // 設(shè)置成員y的值

  void setStartTime(float startTime); // 設(shè)置成員startTime的值

  void setEndTime(float endTime);   // 設(shè)置成員endTime的值

  void setQuantity(float quantity);  // 設(shè)置成員quantity的值

  void setServiceTime(float serviceTime); // 設(shè)置成員serviceTime的值

  void show(); // 顯示顧客節(jié)點(diǎn)信息

private:

  int id;

  float x;

  float y;

  float startTime;

  float endTime;

  float quantity;

  float serviceTime;

};

#endif

OK,那么現(xiàn)在開(kāi)始貼一下main.cpp代碼(Customer.cpp比較簡(jiǎn)單,就不貼了)

// main.cpp

#include "Customer.h"

#include "tinystr.h"

#include "tinyxml.h"

#include<iostream>

#include<vector>

#include<string>

#include<stdlib.h>

#include<iomanip>

 

using namespace std;

static const int NUM_OF_CUSTOMER = 51;    //顧客數(shù)量

static const char* FILENAME = "RC101_050.xml"; //文件名

 

int main(){

  vector<Customer *> customerSet(0); // 顧客集,每個(gè)元素是Customer對(duì)象的指針

  int i,j,k,count;

  int temp1;  // 存放整型數(shù)據(jù)

  float temp2; // 存放浮點(diǎn)型數(shù)據(jù)

  Customer* customer;  // 臨時(shí)顧客節(jié)點(diǎn)指針

  for (i=0; i<NUM_OF_CUSTOMER; i++) { // 先初始化顧客集

    customer = new Customer();

    customerSet.push_back(customer);

  }

  TiXmlDocument doc(FILENAME);  // 讀入XML文件

  if(!doc.LoadFile()) return -1; // 如果無(wú)法讀取文件,則返回

  TiXmlHandle hDoc(&doc);     // hDoc是&doc指向的對(duì)象

  TiXmlElement* pElem;      // 指向元素的指針

  pElem = hDoc.FirstChildElement().Element(); //指向根節(jié)點(diǎn)

  TiXmlHandle hRoot(pElem);    // hRoot是根節(jié)點(diǎn)

 

  // 讀取x,y,它們放在network->nodes->node節(jié)點(diǎn)中

  TiXmlElement* nodeElem = hRoot.FirstChild("network").FirstChild("nodes").FirstChild("node").Element(); //當(dāng)前指向了node節(jié)點(diǎn)

  count = 0; // 記錄移動(dòng)到了哪個(gè)node節(jié)點(diǎn),并且把該node節(jié)點(diǎn)的信息錄入到順序?qū)?yīng)的customer中

  for(nodeElem; nodeElem; nodeElem = nodeElem->NextSiblingElement()) { // 挨個(gè)讀取node節(jié)點(diǎn)的信息

    customer = customerSet[count]; // 當(dāng)前顧客節(jié)點(diǎn),注意不能賦值給一個(gè)新的對(duì)象,否則會(huì)調(diào)用復(fù)制構(gòu)造函數(shù)

    TiXmlHandle node(nodeElem); // nodeElem所指向的節(jié)點(diǎn)

    TiXmlElement* xElem = node.FirstChild("cx").Element(); // cx節(jié)點(diǎn)

    TiXmlElement* yElem = node.FirstChild("cy").Element(); // cy節(jié)點(diǎn)

    nodeElem->QueryIntAttribute("id", &temp1); //把id放到temp1中,屬性值讀法

    customer->setId(temp1);     

    temp2 = atof(xElem->GetText());  // char轉(zhuǎn)float

    customer->setX(temp2);

    temp2 = atof(yElem->GetText());

    customer->setY(temp2);

    count++;

  }

 

  // 讀取其余信息

  TiXmlElement* requestElem = hRoot.FirstChild("requests").FirstChild("request").Element(); // 指向了request節(jié)點(diǎn)

  count = 1;

  for(requestElem; requestElem; requestElem = requestElem->NextSiblingElement()) {

    customer = customerSet[count];   // 當(dāng)前顧客節(jié)點(diǎn),注意不能賦值給一個(gè)新的對(duì)象,否則會(huì)調(diào)用復(fù)制構(gòu)造函數(shù)

    TiXmlHandle request(requestElem); // 指針指向的對(duì)象

    TiXmlElement* startTimeElem = request.FirstChild("tw").FirstChild("start").Element(); // start time

    TiXmlElement* endTimeElem = request.FirstChild("tw").FirstChild("end").Element();   // end time

    TiXmlElement* quantityElem = request.FirstChild("quantity").Element();        // quantity

    TiXmlElement* serviceTimeElem = request.FirstChild("service_time").Element();     // service time

    // 分別讀取各項(xiàng)數(shù)據(jù)

    temp2 = atof(startTimeElem->GetText());

    customer->setStartTime(temp2); 

    temp2 = atof(endTimeElem->GetText());

    customer->setEndTime(temp2);

    temp2 = atof(quantityElem->GetText());

    customer->setQuantity(temp2);

    temp2 = atof(serviceTimeElem->GetText());

    customer->setServiceTime(temp2);

    count++;

  }

 

  // 將讀取到的信息輸出到控制臺(tái)

  cout<<setiosflags(ios_base::left)<<setw(6)<<"id"<<setw(6)<<"x"<<setw(6)<<

    "y"<<setw(12)<<"startTime"<<setw(12)<<"endTime"<<setw(12)<<"quantity"<<setw(14)<<"serviceTime"<<endl;

  for(i=0; i<NUM_OF_CUSTOMER; i++) {

    customer = customerSet[i];

    customer->show();

  }

  system("pause");

  return 0;

}

在解釋main.cpp的內(nèi)容之前,先解釋一下一些數(shù)據(jù)類型(只是個(gè)人理解,歡迎糾錯(cuò)):

  • TiXmlDocument:文件節(jié)點(diǎn),把Xml文件的內(nèi)容讀入到該類型變量中
  • TiXmlElement*:指向節(jié)點(diǎn)的指針
  • TiXmlHandle:節(jié)點(diǎn)的實(shí)例,也就是TiXmlElement所指向的對(duì)象
  • FirstChild("nodeName"):第一個(gè)名字為“nodeName”的子節(jié)點(diǎn)
  • NextSiblingElement():下一個(gè)兄弟節(jié)點(diǎn)元素,它們有相同的父節(jié)點(diǎn)
  • QueryIntAttribute("attributeName", &var):把節(jié)點(diǎn)屬性名為attributeName的屬性值以int類型賦值給var變量
  • GetText():獲取當(dāng)前節(jié)點(diǎn)元素的內(nèi)容,即包含在<node>text</node>中的text

OK,有了以上一些簡(jiǎn)單的知識(shí)積累,就可以很方便地讀取Xml文件了,現(xiàn)在截取xml的部分來(lái)講解:

<instance>

  <network>

    <nodes>

      <node id="0" type="0">

        <cx>40.0</cx>

        <cy>50.0</cy>

      </node>

    <!-- 有N+1個(gè)這樣的node節(jié)點(diǎn) -->

    </nodes>  

  </network>

  .....

</instance>

在這部分我們會(huì)把顧客的id,坐標(biāo)x,y都讀入到Customer對(duì)象中?!?/p>

1. 首先我們得到了文件節(jié)點(diǎn)hDoc,現(xiàn)在我們要進(jìn)入根節(jié)點(diǎn)"instance":

TiXmlElement* pElem;      // 指向元素的指針

pElem = hDoc.FirstChildElement().Element(); //指向根節(jié)點(diǎn)

TiXmlHandle hRoot(pElem);    // hRoot是根節(jié)點(diǎn)

根節(jié)點(diǎn)"instance"是文件節(jié)點(diǎn)的第一個(gè)子節(jié)點(diǎn),所以用 pElem = hDoc.FirstChildElement().Element() 就可以使得指針pElem指向"instance",hRoot是pElem所指向的對(duì)象。

2. 現(xiàn)在我們需要進(jìn)入到“node”節(jié)點(diǎn)中,遍歷其兄弟節(jié)點(diǎn),將所有數(shù)據(jù)讀入。下面的語(yǔ)句可以將第一個(gè)“node”節(jié)點(diǎn)的指針賦值給nodeElem:

TiXmlElement* nodeElem = hRoot.FirstChild("network").FirstChild("nodes").FirstChild("node").Element(); //當(dāng)前指向了node節(jié)點(diǎn)

節(jié)點(diǎn)的id值放在"node"節(jié)點(diǎn)的屬性"id"中:

nodeElem->QueryIntAttribute("id", &temp1); //把id放到temp1中,屬性值讀法

然后坐標(biāo)x, y的值放在“node”節(jié)點(diǎn)的子節(jié)點(diǎn)"cx"和"cy"的內(nèi)容(text)中,所以我們這樣來(lái)讀取:

TiXmlElement* xElem = node.FirstChild("cx").Element(); // cx節(jié)點(diǎn)

temp2 = atof(xElem->GetText());  // char轉(zhuǎn)float

函數(shù)atof在庫(kù)<stdlib>中,用以將char數(shù)組轉(zhuǎn)化為浮點(diǎn)數(shù)。

通過(guò)1,2兩步,我們已經(jīng)把第一個(gè)“node”節(jié)點(diǎn)的id, x, y的值讀入到對(duì)象中,然后只需要把遍歷所有的兄弟節(jié)點(diǎn)即可:

for(nodeElem; nodeElem; nodeElem = nodeElem->NextSiblingElement()) {

......

}

讀入requests節(jié)點(diǎn)下的startTime, endTime, quantity, serviceTime等值的方法也是一樣的,詳情參考main.cpp代碼。

運(yùn)行結(jié)果如下:

總結(jié):

其實(shí)說(shuō)白了讀取Xml文件的關(guān)鍵在于:

  • 移動(dòng)指針到所要讀取數(shù)據(jù)的節(jié)點(diǎn)中; 
  • 若是讀取屬性值,則使用QueryIntAttribute方法直接讀??;
  • 若讀取的是節(jié)點(diǎn)的內(nèi)容,則使用getText()方法讀取;
  • 連續(xù)的數(shù)據(jù)具有兄弟節(jié)點(diǎn)關(guān)系,使用NextSiblingElement()方法來(lái)指向下一個(gè)兄弟節(jié)點(diǎn)

 后記:

這篇博文只介紹了如何讀取Xml文件,至于如何寫入Xml文件,請(qǐng)參考tinyXml的官方教程,講的特別清楚,特別良心。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • C/C++獲取當(dāng)前時(shí)間的方法總結(jié)(最全)

    C/C++獲取當(dāng)前時(shí)間的方法總結(jié)(最全)

    這篇文章主要為大家整理了C/C++中獲取當(dāng)前時(shí)間的最全方法,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)和借鑒價(jià)值,需要的可以了解一下
    2023-03-03
  • 使用C語(yǔ)言實(shí)現(xiàn)掃雷游戲

    使用C語(yǔ)言實(shí)現(xiàn)掃雷游戲

    這篇文章主要為大家詳細(xì)介紹了使用C語(yǔ)言實(shí)現(xiàn)掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • C語(yǔ)言由淺入深了解變量的應(yīng)用

    C語(yǔ)言由淺入深了解變量的應(yīng)用

    這篇文章主要介紹了C語(yǔ)言的變量,變量是C語(yǔ)言語(yǔ)法和語(yǔ)義中一個(gè)很重要的知識(shí)點(diǎn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-05-05
  • 淺析操作系統(tǒng)中的虛擬地址與物理地址

    淺析操作系統(tǒng)中的虛擬地址與物理地址

    本文主要介紹了操作系統(tǒng)中的虛擬地址與物理地址。在早期的計(jì)算機(jī)中,要運(yùn)行一個(gè)程序,會(huì)把這些程序全都裝入內(nèi)存,程序都是直接運(yùn)行在內(nèi)存上的,也就是說(shuō)程序中訪問(wèn)的內(nèi)存地址都是實(shí)際的物理內(nèi)存地址。那當(dāng)程序同時(shí)運(yùn)行多個(gè)程序時(shí),操作系統(tǒng)是如何為這些程序分配內(nèi)存的呢
    2021-06-06
  • C++實(shí)現(xiàn)計(jì)算器功能

    C++實(shí)現(xiàn)計(jì)算器功能

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)計(jì)算器功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • 從c++標(biāo)準(zhǔn)庫(kù)指針萃取器談一下traits技法(推薦)

    從c++標(biāo)準(zhǔn)庫(kù)指針萃取器談一下traits技法(推薦)

    本篇文章基于gcc中標(biāo)準(zhǔn)庫(kù)源碼剖析一下標(biāo)準(zhǔn)庫(kù)中的模板類pointer_traits,并且以此為例理解一下traits技法,對(duì)c++ traits技法源碼分析感興趣的朋友跟隨小編一起看看吧
    2021-07-07
  • C++基礎(chǔ)入門教程(三):數(shù)組、字符串、結(jié)構(gòu)體、共用體

    C++基礎(chǔ)入門教程(三):數(shù)組、字符串、結(jié)構(gòu)體、共用體

    這篇文章主要介紹了C++基礎(chǔ)入門教程(三):數(shù)組、字符串、結(jié)構(gòu)體、共用體,需要的朋友可以參考下
    2014-11-11
  • C++特殊類設(shè)計(jì)概念與示例講解

    C++特殊類設(shè)計(jì)概念與示例講解

    本文介紹C++中三種特殊類設(shè)計(jì)模式:?jiǎn)卫J?、工廠模式和代理模式。通過(guò)詳細(xì)講解每種設(shè)計(jì)模式的實(shí)現(xiàn)原理和應(yīng)用場(chǎng)景,幫助讀者理解和掌握這些常用的面向?qū)ο笤O(shè)計(jì)模式,并提供示例代碼和技巧,便于實(shí)際應(yīng)用
    2023-04-04
  • C++實(shí)現(xiàn)關(guān)機(jī)功能詳細(xì)代碼

    C++實(shí)現(xiàn)關(guān)機(jī)功能詳細(xì)代碼

    大家好,本篇文章主要講的是C++實(shí)現(xiàn)關(guān)機(jī)功能詳細(xì)代碼,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下
    2022-01-01
  • 深入解析C++中的動(dòng)態(tài)類型轉(zhuǎn)換與靜態(tài)類型轉(zhuǎn)換運(yùn)算符

    深入解析C++中的動(dòng)態(tài)類型轉(zhuǎn)換與靜態(tài)類型轉(zhuǎn)換運(yùn)算符

    這篇文章主要介紹了C++中的動(dòng)態(tài)類型轉(zhuǎn)換與靜態(tài)類型轉(zhuǎn)換運(yùn)算符,即dynamic_cast與static_cast的用法,需要的朋友可以參考下
    2016-01-01

最新評(píng)論