使用YAML-CPP将

时间:2018-11-07 22:07:48

标签: java c++ stl yaml yaml-cpp

我有一个 yaml 文件,其中包含嵌套的地图:

SOLAR-SYSTEM:
  my/planet:
    earth: blue
  my/satellite:
    moon: white

我正在使用 yaml-cpp 解析这些值。

有什么方法可以从yaml文件中提取这些值并将其添加到 stl映射

这将我带到问题的第二部分。

我对C ++还是很陌生,所以不完全了解地图在其中的工作方式。

在Java中,我使用 snakeyaml 解析相同的yaml文件。

它将地图添加到三元哈希图数据结构中:

HashMap<String, Map<String, Map<String, String>>>

我可以方便地尝试一下。

在C ++中,有什么简单的方法可以做到这一点吗?

2 个答案:

答案 0 :(得分:1)

我已经将此大型库用于YAML解析器2天了。因此,我可能还会犯一些错误。我使用yaml-cpp ver0.6.2。

主要思想是构建自己的结构。 之后,将模板专业化用于特定类型的转换。

我认为您的文档结构不是很好。感觉就像嵌套std :: map一样。我想您可以看一下示例Only for yaml file,因为这些API很旧

最后,您可以将值拉入您构建的结构中。

对不起,我的英语不好。所以给你看我的代码。您可以再问我是否有新问题。

此外,代码作者在这里。您可能会从他那里得到更准确的答案。

struct Planet {
    std::string earth;
};

struct Satellite {
    std::string moon;
};

struct SolarSystem {
    Planet p;
    Satellite s;
};

namespace YAML {
template<>
struct convert<Planet> {
    static Node encode(const Planet &rhs) {
        Node node;
        node["earth"] = rhs.earth;
        return node;
    }

    static bool decode(const Node &node, Planet &rhs) {
        if (!node.IsMap())
            return false;
        rhs.earth = node["earth"].as<std::string>();
        return true;
    }
};

template<>
struct convert<Satellite> {
    static Node encode(const Satellite &rhs) {
        Node node;
        node["moon"] = rhs.moon;
        return node;
    }

    static bool decode(const Node &node, Satellite &rhs) {
        if (!node.IsMap())
            return false;
        rhs.moon = node["moon"].as<std::string>();
        return true;
    }
};

template<>
struct convert<SolarSystem> {
    static Node encode(const SolarSystem &rhs) {
        Node node;
        node["my/planet"] = rhs.p;
        node["my/satellite"] = rhs.s;
        return node;
    }

    static bool decode(const Node &node, SolarSystem &rhs) {
        if (!node.IsMap())
            return false;

        rhs.p = node["my/planet"].as<Planet>();
        rhs.s = node["my/satellite"].as<Satellite>();
        return true;
    }
};
}

int main(void)
{
    YAML::Node doc = YAML::LoadFile("path/to/your/file");
    SolarSystem ss = doc["SOLAR-SYSTEM"].as<SolarSystem>();

    std::cout << ss.p.earth << std::endl;      // "blue"
    std::cout << ss.s.moon << std::endl;       // "white"

    return 0;
}

答案 1 :(得分:0)

不确定为什么被否决了。感谢@Crow的答案。我无法使用它,因为它涉及对值进行硬编码。这是我设计的解决方案:

try {
        YAML::Node firstTierNode = YAML::LoadFile("config.yml");

        std::map<std::string, std::map<std::string, std::map<std::string, std::string>>> newMap1;
        for (YAML::const_iterator it = firstTierNode.begin(); it != firstTierNode.end(); ++it) {

            string level1First = it->first.as<std::string>();

            YAML::Node secondTierNode = it->second;

            std::map<std::string, std::map<std::string, std::string>>  newMap2;
            for (YAML::const_iterator it = secondTierNode.begin(); it != secondTierNode.end(); ++it) {

                string level2First = it->first.as<std::string>();

                YAML::Node thirdTierNode = it->second;

                std::map<std::string, std::string> newMap3;
                for (YAML::const_iterator it = thirdTierNode.begin(); it != thirdTierNode.end(); ++it) {

                    string level3First = it->first.as<std::string>();

                    string level3SecondString = it->second.as<std::string>();

                    newMap3.insert(std::pair<std::string, std::string>(level3First, level3SecondString));
                }

                newMap2.insert(std::pair<std::string, std::map<string, string>>(level2First, newMap3));

            }

            newMap1.insert(std::pair<std::string, std::map<string, std::map<string, string>>>(level1First, newMap2));

        }

        for (const auto& x : newMap1) {
            std::cout <<x.first << endl << endl;
            for (const auto& y : x.second) {
                std::cout << y.first << endl << endl;
                for (const auto& z : y.second) {
                    std::cout << z.first << endl << z.second << endl << endl << endl;
                }
            }
        }

        return 1;
    }
    catch (exception& e) {
            cerr <<e.what()<< endl;
    }