2012-08-12 212 views
1

我有一個問題,我認爲解決方案可能很簡單,但即時卡住。我有一些配置文件,即時通訊設法解析在c + +獲得一些重要的價值。從字符串中提取值C++

它看起來像這樣:

信息大小= 32粗體= 0斜體= 0的unicode = 1 stretchH = 100光滑= 1節AA = 1個填充= 0,0,0,0
共同 lineHeight是= 32鹼= 27 scaleW = 1024 scaleH = 28頁= 1填充= 0 alphaChnl = 1
字符計數= 74
ID = 32 X = 837 Y = 15寬度= 3高度= 1 xoffset = -1 yoffset = 31 xadvance = 8 page = 0 CHNL = 15
ID = 33 X = 802 Y = 0寬度= 4高度= 19 X偏移= 2 Y偏移= 8 xadvance = 8頁= 0 CHNL = 15
ID = 35 X = 292ÿ = 0 width = 17 height = 19 xoffset = 0 yoffset = 8 xadvance = 16 page = 0 chnl = 15
char id = 37 x = 177 y = 0 width = 19 height = 19 xoffset = -1 yoffset = 8 xadvance = 17頁= 0 CHNL = 15
ID = 38×216 = y = 0的寬度= 18高度= 19 X偏移= 0 Y偏移= 8 xadvance = 18頁= 0 CHNL = 15

信息,通用和字符的基本/全球值。每個字符行應該保存在一個類似格式(x,y,height,offsetX,offsetY ...)的結構的數組(或向量)中。

現在我試過getline()一行一行,然後用這些行做一個istringstream來「搜索」這些行以獲取我需要的值。 值得注意的是,IAM不需要所有這些值,我需要一種方法來保存我需要的每一行。

在此先感謝您的幫助!

回答

2

每一行都是以對象的類型爲前綴。
所以你的讀者應該讀第一個字,然後決定要讀取什麼對象。

std::ifstream file("data.txt"); 
std::string type; 
while(file >> type) 
{ 
     if (type == "info") { readInfo(file);} 
    else if (type == "common") { readCommon(file);} 
    else if (type == "chars") { readChars(file);} 
    else if (type == "char") { readChar(file);} 
} 

對於每種類型的對象,您需要定義一個結構來保存數據。

// char id=32 x=837 y=15 width=3 height=1 xoffset=-1 yoffset=31 xadvance=8 page=0 chnl=15 
struct CharData 
{ 
    int id; 
    int x; 
    int y; 
    int width; 
    int height; 
    int xoffset; 
    int yoffset; 
    int xadvance; 
    int page; 
    int chnl; 
}; 

現在你必須定義一個讀取數據的方法。在C++中,我們使用operator>>來從流中讀取數據。

std::istream& operator>>(std::istream& stream, CharData& data) 
{ 
    // All the data is on one line. 
    // So read the whole line (including the '\n') 
    std::string  line; 
    std::getline(stream, line); 

    // convert the single line into a stream for parsing. 
    // One line one object just makes it easier to handle errors this way. 
    std::stringstream linestream(line); 

    // Assume the prefix type information has already been read (now looks like this) 
    // id=32 x=837 y=15 width=3 height=1 xoffset=-1 yoffset=31 xadvance=8 page=0 chnl=15 
    std::string command; 
    while(linestream >> command) // This reads one space separated command from the line. 
    { 
      if (command.substr(0,3) == "id=")  {data.id  = atoi(&command[3]);} 
     else if (command.substr(0,2) == "x=")   {data.x  = atoi(&command[2]);} 
     else if (command.substr(0,2) == "y=")   {data.y  = atoi(&command[2]);} 
     else if (command.substr(0,6) == "width=")  {data.width = atoi(&command[6]);} 
     else if (command.substr(0,7) == "height=") {data.height = atoi(&command[7]);} 
     else if (command.substr(0,8) == "xoffset=") {data.xoffset = atoi(&command[8]);} 
     else if (command.substr(0,8) == "yoffset=") {data.yoffset = atoi(&command[8]);} 
     else if (command.substr(0,9) == "xadvance=") {data.xadvance = atoi(&command[9]);} 
     else if (command.substr(0,5) == "page=")  {data.page  = atoi(&command[5]);} 
     else if (command.substr(0,5) == "chnl=")  {data.chnl  = atoi(&command[5]);} 
    } 
    return stream; 
} 

重複您需要閱讀的其他類型的過程。然後寫入讀取命令變得簡單:

std::vector<CharData> charVector; 
void readChar(std::istream& stream) 
{ 
    CharData  data; 
    stream >> data;    // read the object from the stream 
            // This uses the `operator>>` we just defined above. 

    charVector.push_back(data); // put the data item into a vector. 
} 

重複其他類型的過程。

+0

不應該操作員只讀取類型?其餘的變量應該在依賴於類型的函數中讀取 – Gir 2012-08-12 16:00:18

+0

感謝您的擴展代碼! – puelo 2012-08-12 20:10:43

+0

輕微明顯的錯字:除非我誤解了某些內容,否則上面所有'itoa'的實例都是「atoi」。 – 2012-08-12 20:42:17

0

important_value=str_from_getline.find(what_to_find); 

important_value是size_t,並且有你在函數getline所追求的()字符串的起點

0

嘗試使用的strtok或C++的正則表達式庫

+0

不,不。 strtok()是C的憎惡。在C++中有更好的技術。 – 2012-08-12 15:58:06

+0

你能解釋爲什麼嗎? – Gir 2012-08-12 16:01:02

+0

那麼它使用C-Strings的最糟糕的事情。接下來它改變字符串(所以它不是一個真正的函數)。接下來,這取決於每次調用它時繼續工作的突變。 – 2012-08-12 16:05:06