c++ 解析yaml文件的步驟
作者:li-peng
出處:http://www.cnblogs.com/li-peng/
一直用c++操作ini做配置文件,想換成yaml,在github上搜索,看有沒(méi)有開(kāi)源的庫(kù),功夫不負(fù)有心人,找到了yaml-cpp,用他解析了一個(gè)yaml的例子非常好使,分享一下如何使用他。
先git clone git@github.com:jbeder/yaml-cpp.git下來(lái)編譯成靜態(tài)庫(kù)
mkdir build cd build cmake .. make
運(yùn)行完后,會(huì)得到libyaml-cpp.a。
新建一個(gè)項(xiàng)目,結(jié)構(gòu)大致如下
yaml_demo
|__ include
|__yaml-cpp 頭文件夾
|__ lib
|__yaml-cpp 庫(kù)文件夾
|__ main.cpp配置CMakeLists.txt把頭文件和靜態(tài)庫(kù)加到項(xiàng)目里,這樣在編譯和鏈接時(shí)才能通過(guò)
project(yaml_demo)
set(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin) # 二進(jìn)制文件的輸出目錄
link_directories(${PROJECT_SOURCE_DIR}/lib/yaml-cpp)
add_executable(${PROJECT_NAME} main.cpp)
target_include_directories(${PROJECT_NAME} PUBLIC ${PROJECT_SOURCE_DIR}/include)
target_link_libraries(${PROJECT_NAME} yaml-cpp.a) 對(duì)yaml-cpp的配置就完成了。看一下我的config文件
api: aaaaa v: 1 label: app: hello image: abc containers: - name: abc age: 18 - name: 222 age: 12
其中api和v是比較簡(jiǎn)單的鍵值,我們可以直接讀取他們的值
std::cout << "api: " << config["api"].as<std::string>() << std::endl; std::cout << "v: " << config["v"].as<int>() << std::endl;
label是一個(gè)map,containers是一個(gè)列表,這就要特殊處理一下,yaml-cpp有自己的轉(zhuǎn)換模板
template <typename T> struct convert;
在進(jìn)行轉(zhuǎn)換的時(shí)候他會(huì)判斷有沒(méi)有實(shí)現(xiàn) decode方法
struct as_if<T, void> {
explicit as_if(const Node& node_) : node(node_) {}
const Node& node;
T operator()() const {
if (!node.m_pNode)
throw TypedBadConversion<T>(node.Mark());
T t;
if (convert<T>::decode(node, t))
return t;
throw TypedBadConversion<T>(node.Mark());
}
};Node是yaml-cpp的核心,我們的配置的所有操作都從這個(gè)類(lèi)中進(jìn)行。
我們只要具體化自定義的struct就可以使用了
struct label {
std::string app;
std::string image;
};
namespace YAML {
template<>
struct convert<label> {
static Node encode(const label &rhs) {
Node node;
node.push_back(rhs.app);
node.push_back(rhs.image);
return node;
}
static bool decode(const Node &node, label &rhs) {
std::cout << node.Type() << std::endl;
rhs.app = node["app"].as<std::string>();
rhs.image = node["image"].as<std::string>();
return true;
}
};
}encode方法是把我們自定義的struct轉(zhuǎn)換成yaml-cpp的Node,
轉(zhuǎn)換時(shí)可以這樣
if (config["label"]) {
label l = config["label"].as<label>();
std::cout << "app: " << l.app << " image: " << l.image << std::endl;
}container也是一樣的具體化
struct container {
std::string name;
int age;
};
namespace YAML {
template<>
struct convert<container> {
static Node encode(const container &rhs) {
Node node;
node.push_back(rhs.name);
node.push_back(rhs.age);
return node;
}
static bool decode(const Node &node, container &rhs) {
rhs.name = node["name"].as<std::string>();
rhs.age = node["age"].as<int>();
return true;
}
};
}完整代碼如下:
#include <string>
#include <iostream>
#include <yaml-cpp/yaml.h>
#include <yaml-cpp/node/parse.h>
struct container {
std::string name;
int age;
};
namespace YAML {
template<>
struct convert<container> {
static Node encode(const container &rhs) {
Node node;
node.push_back(rhs.name);
node.push_back(rhs.age);
return node;
}
static bool decode(const Node &node, container &rhs) {
rhs.name = node["name"].as<std::string>();
rhs.age = node["age"].as<int>();
return true;
}
};
}
struct label {
std::string app;
std::string image;
};
namespace YAML {
template<>
struct convert<label> {
static Node encode(const label &rhs) {
Node node;
node.push_back(rhs.app);
node.push_back(rhs.image);
return node;
}
static bool decode(const Node &node, label &rhs) {
std::cout << node.Type() << std::endl;
rhs.app = node["app"].as<std::string>();
rhs.image = node["image"].as<std::string>();
return true;
}
};
}
int main(int argc, char **argv) {
std::string config_path = "./config.yaml";
std::cout << config_path << std::endl;
YAML::Node config = YAML::LoadFile(config_path);
std::cout << "api: " << config["api"].as<std::string>() << std::endl;
std::cout << "v: " << config["v"].as<int>() << std::endl;
if (config["label"]) {
label l = config["label"].as<label>();
std::cout << "app: " << l.app << " image: " << l.image << std::endl;
}
if (config["containers"]) {
std::vector<container> vi = config["containers"].as<std::vector<container>>();
for (std::vector<container>::iterator it = vi.begin(); it != vi.end(); ++it) {
std::cout << "vector: name: " << it->name << " age: " << it->age << std::endl;
}
}
return 0;
}以上就是c++ 解析yaml文件的步驟的詳細(xì)內(nèi)容,更多關(guān)于c++ 解析yaml文件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C語(yǔ)言鏈表實(shí)現(xiàn)工資管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言鏈表實(shí)現(xiàn)工資管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
手把手帶你學(xué)習(xí)C++的數(shù)據(jù)類(lèi)型
這篇文章主要為大家介紹了C++的數(shù)據(jù)類(lèi)型,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助,希望能夠給你帶來(lái)幫助2021-11-11
c++中的volatile和variant關(guān)鍵字詳解
大家好,本篇文章主要講的是c++中的volatile和variant關(guān)鍵字詳解,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話(huà)記得收藏一下2022-01-01
C++自動(dòng)析構(gòu)時(shí)的順序問(wèn)題
這篇文章主要介紹了C++自動(dòng)析構(gòu)時(shí)的順序,通過(guò)實(shí)例代碼給大家講解了C++ 構(gòu)造與析構(gòu)的執(zhí)行順序,代碼簡(jiǎn)單易懂,非常不錯(cuò)對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
C語(yǔ)言完美實(shí)現(xiàn)動(dòng)態(tài)數(shù)組代碼分享
本文給大家分享的是一則使用C語(yǔ)言實(shí)現(xiàn)動(dòng)態(tài)數(shù)組的代碼,完美解決內(nèi)存溢出以及內(nèi)存回收問(wèn)題,有需要的小伙伴可以參考下。2016-02-02
C++ min/max_element 函數(shù)用法詳解
這篇文章主要介紹了C++ min/max_element 函數(shù)用法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-02-02
c++ 如何在libuv中實(shí)現(xiàn)tcp服務(wù)器
這篇文章主要介紹了c++ 如何在libuv中實(shí)現(xiàn)tcp服務(wù)器,幫助大家更好的理解和使用libuv,感興趣的朋友可以了解下2021-02-02
VS2022 無(wú)法打開(kāi)源文件“stdio.h”問(wèn)題解決
本文主要介紹了VS2022 無(wú)法打開(kāi)源文件“stdio.h”問(wèn)題解決,文中通過(guò)圖文的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-06-06

