Boost属性树指定允许的值 我想使用升压属性树来处理我的C++应用程序的设置,因为它在这个场景中被广泛使用。

Boost属性树指定允许的值 我想使用升压属性树来处理我的C++应用程序的设置,因为它在这个场景中被广泛使用。,c++,boost,boost-propertytree,C++,Boost,Boost Propertytree,我的问题:当更改属性树中的值(通过xml解析或手动)时,是否有方法提前指定键的允许值列表? 例如,如果我想做一个简单的“是/否”设置,我是否必须用if-条件检查值,或者我是否可以教我的树预先只接受特定键的两个值“是”和“否”,以便它在出错时抛出异常 您可以为此使用翻译器。我记得有一篇很好的博客文章描述了如何在XML支持的属性树中进行自定义日期格式解析: 让我们以你为例: enum class YesNo { No, Yes }; 在这种情况下,调用代码可能如下所示: static Yes

我的问题:当更改属性树中的值(通过xml解析或手动)时,是否有方法提前指定键的允许值列表?
例如,如果我想做一个简单的“是/否”设置,我是否必须用if-条件检查值,或者我是否可以教我的树预先只接受特定键的两个值“是”和“否”,以便它在出错时抛出异常

您可以为此使用翻译器。我记得有一篇很好的博客文章描述了如何在XML支持的属性树中进行自定义日期格式解析:

让我们以你为例:

enum class YesNo { No, Yes };
在这种情况下,调用代码可能如下所示:

static YesNoTranslator trans;

int main() {

    std::istringstream iss(R"(
            <?xml version="1.0"?>
            <demo>
                <positive>Yes</positive>
                <negative>No</negative>
                <invalid>Bogus</invalid>
            </demo>
        )");

    ptree pt;
    read_xml(iss, pt);


    for (auto&& field : { "demo.positive", "demo.negative", "demo.invalid" })
    {
        try {
            std::cout << "With 'No' default: '" << field << "':\t" << pt.get(field, YesNo::No, trans) << "\n";
            std::cout << "Without default:   '" << field << "':\t" << pt.get<YesNo>(field, trans)     << "\n";
        } catch(std::exception const& e) {
            std::cout << "Error parsing '"      << field << "':\t" << e.what()                        << "\n";
        }
    }
}

在发布我的答案后,我发现已经有涉及属性树的翻译器。也许你在那里也发现了一些有趣的东西。
#include <boost/property_tree/ptree.hpp>
#include <boost/property_tree/xml_parser.hpp>
#include <sstream>
#include <iostream>

using boost::property_tree::ptree;

enum class YesNo { No, Yes };

static inline std::ostream& operator<<(std::ostream& os, YesNo v) {
    switch(v) {
        case YesNo::Yes: return os << "Yes";
        case YesNo::No:  return os << "No";
    }
    return os << "??";
}

struct YesNoTranslator {
    typedef std::string  internal_type;
    typedef YesNo        external_type;

    boost::optional<external_type> get_value(internal_type const& v) {
        if (v == "Yes") return YesNo::Yes;
        if (v == "No")  return YesNo::No;

        return boost::none;
    }

    boost::optional<internal_type> put_value(external_type const& v) {
        switch(v) {
            case YesNo::Yes: return std::string("Yes");
            case YesNo::No:  return std::string("No");
            default: throw std::domain_error("YesNo");
        }
    }
};

static YesNoTranslator trans;

int main() {

    std::istringstream iss(R"(
            <?xml version="1.0"?>
            <demo>
                <positive>Yes</positive>
                <negative>No</negative>
                <invalid>Bogus</invalid>
            </demo>
        )");

    ptree pt;
    read_xml(iss, pt);


    for (auto&& field : { "demo.positive", "demo.negative", "demo.invalid" })
    {
        try {
            std::cout << "With 'No' default: '" << field << "':\t" << pt.get(field, YesNo::No, trans) << "\n";
            std::cout << "Without default:   '" << field << "':\t" << pt.get<YesNo>(field, trans)     << "\n";
        } catch(std::exception const& e) {
            std::cout << "Error parsing '"      << field << "':\t" << e.what()                        << "\n";
        }
    }

}
With 'No' default: 'demo.positive': Yes
Without default:   'demo.positive': Yes
With 'No' default: 'demo.negative': No
Without default:   'demo.negative': No
With 'No' default: 'demo.invalid':  No
Without default:   'demo.invalid':  Error parsing 'demo.invalid':   conversion of data to type "5YesNo" failed