Boost属性树解析自定义配置格式



根据@sehe在这篇文章中提供的Boost_option解析配置文件的链接,我需要解析可能有注释的配置文件。

https://www.boost.org/doc/libs/1_76_0/doc/html/property_tree/parsers.html#property_tree.parsers.info_parser

但是,既然有注释(leading#(,那么除了read_info((之外,还应该使用grammar_spirit来删除注释吗?我指的是/property_tree/examples文件夹中的info_grammar_spirit.cpp

您最好避免依赖于实现细节,因此我建议预处理您的配置文件以去除注释。

用CCD_ 2简单地替换CCD_ 1可能就足够了。

基于之前的答案:

std::string tmp;
{
std::ifstream ifs(file_name.c_str());
tmp.assign(std::istreambuf_iterator<char>(ifs), {});
} // closes file
boost::algorithm::replace_all(tmp, "//", ";");
std::istringstream preprocessed(tmp);
read_info(preprocessed, pt);

现在,如果您更改输入以包含注释:

Resnet50 {
Layer CONV1 {
Type: CONV // this is a comment
Stride { X: 2, Y: 2 }       ; this too
Dimensions { K: 64, C: 3, R: 7, S: 7, Y:224, X:224 }
}
// don't forget the CONV2_1_1 layer
Layer CONV2_1_1 {
Type: CONV
Stride { X: 1, Y: 1 }       
Dimensions { K: 64, C: 64, R: 1, S: 1, Y: 56, X: 56 }
}
}

如果我们还扩展调试输出以验证:,它仍然可以按预期进行解析

ptree const& resnet50 = pt.get_child("Resnet50");
for (auto& entry : resnet50) {
std::cout << entry.first << " " << entry.second.get_value("") << "n";
std::cout << " --- Echoing the complete subtree:n";
write_info(std::cout, entry.second);
}

打印

Layer CONV1
--- Echoing the complete subtree:
Type: CONV
Stride
{
X: 2,
Y: 2
}
Dimensions
{
K: 64,
C: 3,
R: 7,
S: 7,
Y:224, X:224
}
Layer CONV2_1_1
--- Echoing the complete subtree:
Type: CONV
Stride
{
X: 1,
Y: 1
}
Dimensions
{
K: 64,
C: 64,
R: 1,
S: 1,
Y: 56,
X: 56
}

查看Coliru直播

是的,但是

如果字符串文字中出现了"//",该怎么办?它不会也被替换吗。对

这不是一个图书馆质量的解决方案。您不应该期望有一个,因为您不必付出任何努力来解析定制的配置文件格式。

你是唯一一个能够判断这种方法的缺点是否对你来说是个问题的人。

然而,除了复制和修改Boost的解析器或从头开始实现自己的解析器之外,没有什么可做的

对于受虐狂

如果你不想重新实现整个解析器,但仍然想要";智能;要跳过字符串文字,这里有一个pre_process函数可以完成所有这些操作。这一次,它真正采用了Boost Spirit

#include <boost/spirit/home/x3.hpp>
std::string pre_process(std::string const& input) {
std::string result;
using namespace boost::spirit::x3;
auto static string_literal
= raw[ '"' >> *('\'>> char_ | ~char_('"')) >> '"' ];
auto static comment
= char_(';') >> *~char_("rn")
| "//" >> attr(';') >> *~char_("rn")
| omit["/*" >> *(char_ - "*/") >> "*/"];
auto static other
= +(~char_(";"") - "//" - "/*");
auto static content
= *(string_literal | comment | other) >> eoi;
if (!parse(begin(input), end(input), content, result)) {
throw std::invalid_argument("pre_process");
}
return result;
}

正如您所看到的,它识别字符串文字(带有转义符(,它处理"/"one_answers";"将线条注释设置为等效样式。致";炫耀;我抛出了/块注释/,这些注释无法用正确的INFO语法表示,所以我们只omit[]它们。

现在让我们用一个时髦的例子来测试(从文档中的"演示所有INFO功能的复杂例子"扩展而来(:

#include <boost/property_tree/info_parser.hpp>
#include <iostream>
using boost::property_tree::ptree;
int main() {
boost::property_tree::ptree pt;
std::istringstream iss(
pre_process(R"~~( ; A comment
key1 value1   // Another comment
key2 "value with /* no problem */ special // characters in it {};#nt""
{
subkey "value split "
"over three"
"lines"
{
a_key_without_value ""
"a key with special characters in it {};#nt"" ""
"" value    /* Empty key with a value */
"" /*also empty value: */ ""       ; Empty key with empty value!
}
})~~"));
read_info(iss, pt);
std::cout << " --- Echoing the parsed tree:n";
write_info(std::cout, pt);
}

打印(Coliru直播(

--- Echoing the parsed tree:
key1 value1
key2 "value with /* no problem */ special // characters in it {};#n    ""
{
subkey "value split over threelines"
{
a_key_without_value ""
"a key with special characters in it {};#n     "" ""
"" value
"" ""
}
}

相关内容

  • 没有找到相关文章

最新更新