一直用c++操作ini做配置文件,想换成yaml,在全球最大的同性交友网站github上搜索,看有没有开源的库,功夫不负有心人,找到了yaml-cpp,用他解析了一个yaml的例子非常好使,分享一下如何使用他。

git clone git@github.com:jbeder/yaml-cpp.git下来编译成静态库

mkdir build
cd build
cmake ..
make

运行完后,会得到libyaml-cpp.a

新建一个项目,结构大致如下

yaml_demo
|__ include
|__yaml-cpp 头文件夹
|__ lib
|__yaml-cpp 库文件夹
|__ main.cpp

配置CMakeLists.txt把头文件和静态库加到项目里,这样在编译和链接时才能通过

project(yaml_demo)
set(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin) # 二进制文件的输出目录
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)

yaml-cpp的配置就完成了。看一下我的config文件

api: aaaaa
v: 1
label:
app: hello
image: abc
containers:
- name: abc
age: 18
- name: 222
age: 12

其中apiv是比较简单的键值,我们可以直接读取他们的值

    std::cout << "api: " << config["api"].as<std::string>() << std::endl;
std::cout << "v: " << config["v"].as<int>() << std::endl;

label是一个mapcontainers是一个列表,这就要特殊处理一下,yaml-cpp有自己的转换模板

template <typename T>
struct convert;

在进行转换的时候他会判断有没有实现 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());
}
};

Nodeyaml-cpp的核心,我们的配置的所有操作都从这个类中进行。

我们只要具体化自定义的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转换成yaml-cppNode

转换时可以这样

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文件的更多相关文章

  1. 使用ruamel.yaml库,解析yaml文件

    在实现的需求如下: 同事提供了一个文本文件,内含200多个host与ip的对应关系,希望能在k8s生成pod时,将这些对应关系注入到/etc/hosts中. 网上看文档,这可以通过扩充pod中的hos ...

  2. python解析yaml文件

    YAML语法规则: http://www.ibm.com/developerworks/cn/xml/x-cn-yamlintro/ 下载PyYAML: http://www.yaml.org/ 解压 ...

  3. 解析YAML文件

    YamlMapFactoryBean yamlMapFactoryBean = new YamlMapFactoryBean(); yamlMapFactoryBean.setResources(ne ...

  4. python 解析 yaml文件

    import yaml with open("./test.yaml") as f: x = yaml.load(f) print(x) [{'tasks': [{'yum': { ...

  5. Python--代码1(接口测试:测试用例从数据库读取写到yaml文件中)

    一. 从数据库中读取全部接口,并写入yaml文件 数据库中的数据存储格式如下图: import pymysql import os import json # from ruamel import y ...

  6. YAML文件解析

    YAML是“另一种标记语言”的外语缩写,YAML 是一种比JSON(json多层次{ 与 [ 会被搞晕的)更直观的表现形式,展示上更易查错和关系描述.因为不需要一个专业工具就可以排查正确性.YAML目 ...

  7. python基础——python解析yaml类型文件

    一.yaml介绍 yaml全称Yet Another Markup Language(另一种标记语言).采用yaml作为配置文件,文件看起来直观.简洁.方便理解.yaml文件可以解析字典.列表和一些基 ...

  8. springboot中对yaml文件的解析

    一.YAML是“YAML不是一种标记语言”的外语缩写 (见前方参考资料原文内容):但为了强调这种语言以数据做为中心,而不是以置标语言为重点,而用返璞词重新命名.它是一种直观的能够被电脑识别的数据序列化 ...

  9. yaml文件解析详解

    前言 yaml文件是什么?yaml文件其实也是一种配置文件类型,相比较ini,conf配置文件来说,更加的简洁,操作也更加简单,同时可以存放不同类型的数据,不会改变原有数据类型,所有的数据类型在读取时 ...

随机推荐

  1. Android10_原理机制系列_AMS之AMS的启动

    概述 该篇基于AndroidQ,主要介绍系统启动中的 AMS(ActivityManagerService)的启动过程. AMS对四大组件(AndroidQ将activity移到了ActivityTa ...

  2. SQL server分页的四种方法(算很全面了)

      这篇博客讲的是SQL server的分页方法,用的SQL server 2012版本.下面都用pageIndex表示页数,pageSize表示一页包含的记录.并且下面涉及到具体例子的,设定查询第2 ...

  3. NOIP2015 解题报告

    Day1 T3 运输计划 二分之后做一遍树上差分,找出被所有时间超限的运输计划覆盖的花费时间最长的航道,将其改造成虫洞. LCA 用倍增求可能会被卡常,建议用 Tarjan 求.

  4. Java反射——java.lang.Class和类的加载

    反射的基础: java.lang.Class Class类的实例对象,用于记录类描述信息. 源码说:represent classes and interfaces in a running Java ...

  5. Java中的单例模式最全解析

    单例模式是 Java 中最简单的设计模式之一,它是指一个类在运行期间始终只有一个实例,我们就把它称之为单例模式.它不但被应用在实际的工作中,而且还是面试中最常考的题目之一.通过单例模式我们可以知道此人 ...

  6. Spring与Web环境集成

    1. Spring与Web环境集成 1.1 ApplicationContext应用上下文获取方式 应用上下文对象是通过new ClasspathXmlApplicationContext(sprin ...

  7. Docker 指南

    一.docker 介绍 1.1 引言 环境不一致 "我本地运行没问题啊?!" 多用户相互影响 "哪个哥们又写死循环了,怎么这么卡?!" 运维成本高 " ...

  8. 跟我一起学Redis之Redis持久化必知必会

    前言 Redis是出了名的速度快,那是因为在内存中进行数据存储和操作:如果仅仅是在内存中进行数据存储,那就会导致以下问题: 数据随进程退出而消失:当服务器断电或Redis Server进程退出时,内存 ...

  9. 浅尝 Elastic Stack (五) Logstash + Beats + Kafka

    在 Elasticsearch.Kibana.Beats 安装 中讲到推荐架构: 本文基于 Logstash + Beats 读取 Spring Boot 日志 将其改为上述架构 如果没有安装 Kaf ...

  10. Flutter——Dart Extension扩展方法的使用

    dart的extension方法可以给已经存在的类添加新的函数,通过extension我们可以封装一些常用方法,提高开发效率. 例一:扩展String 给string添加一个log打印方法 exten ...