- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我目前正在尝试使用 boost::spirit::qi::phrase_parse
完成一些工作但我无法自己解决这个问题。
值得一提:我对 boost 和 boost::spirit 是全新的。
我收到了 "{A [B C] -> F [D E], C ->E,B->Z}"
形式的输入
我想将这种类型的输入解析为 std::map<std::string, std::string>
. key 应包含每个 std::string
在 "->"
之前和每个 std::string
的值在"->"
之后直到 ','
发生。
此外 '['
和 ']'
不应存储。
所以std::map
的内容解析成功后应该是这样的:
{
("A", "F"),
("A", "D E"),
("B C", "F"),
("B C", "D E"),
("C", "E"),
("B", "Z")
}
我的第一种方法是将所有键/值存储在 std::vector<std::string>
中.
#include <boost/spirit/include/qi.hpp>
#include <iostream>
#include <string>
#include <vector>
int main()
{
using boost::spirit::qi::phrase_parse;
using boost::spirit::qi::char_;
using boost::spirit::qi::lexeme;
std::string input = "{A [B C] -> F [D E], C ->E,B->Z}";
std::string::const_iterator beg = input.begin(), end = input.end();
std::vector<std::string> sdvec;
bool r = phrase_parse( beg,
end,
'{' >> (+(+char_("a-zA-Z0-9") | lexeme[('[' >> +char_("a-zA-Z0-9 ") >> ']')]) >> '-' >> '>' >> +(+char_("a-zA-Z0-9") | lexeme[('[' >> +char_("a-zA-Z0-9 ") >> ']')])) % ',' >> '}',
boost::spirit::ascii::space,
sdvec
);
if(beg != end) {
std::cout << "Parsing failed!" << std::endl;
} else {
std::cout << "Parsing succeeded!" << std::endl;
}
for(int i=0; i<sdvec.size(); i++) {
std::cout << i << ": " << sdvec[i] << std::endl;
}
return 0;
}
执行这个我找到了每个 std::string
作为 std::vector
的条目:
Parsing 2 succeeded!
0: A
1: B C
2: F
3: D E
4: C
5: E
6: B
7: Z
但我不知道如何将这些值解析为 std::map<std::string, std::string>
使用 boost::spirit::qi::phrase_parse
因为简单地替换会抛出一些编译错误。
编辑:
实际上我找到了我需要的东西:http://boost-spirit.com/home/articles/qi-example/parsing-a-list-of-key-value-pairs-using-spirit-qi/
我根据自己的问题采用了本文的代码:
#include <boost/spirit/include/qi.hpp>
#include <boost/fusion/include/std_pair.hpp>
#include <iostream>
#include <string>
#include <vector>
#include <map>
namespace qi = boost::spirit::qi;
template <typename Iterator>
struct keys_and_values
: qi::grammar<Iterator, std::map<std::string, std::string>()>
{
keys_and_values()
: keys_and_values::base_type(query)
{
query = '{' >> *qi::lit(' ') >> pair >> *(qi::lit(',') >> *qi::lit(' ') >> pair) >> *qi::lit(' ') >> '}';
pair = key >> -(*qi::lit(' ') >> "->" >> *qi::lit(' ') >> value);
key = +qi::char_("a-zA-Z0-9") | qi::lexeme[('[' >> +qi::char_("a-zA-Z0-9 ") >> ']')];
value = +qi::char_("a-zA-Z0-9") | qi::lexeme[('[' >> +qi::char_("a-zA-Z0-9 ") >> ']')];
}
qi::rule<Iterator, std::map<std::string, std::string>()> query;
qi::rule<Iterator, std::pair<std::string, std::string>()> pair;
qi::rule<Iterator, std::string()> key, value;
};
int main()
{
std::string input = "{AB -> CD, E -> F, G -> HI, [J K L] -> [M N O] }";
std::string::iterator begin = input.begin();
std::string::iterator end = input.end();
keys_and_values<std::string::iterator> p; // create instance of parser
std::map<std::string, std::string> m; // map to receive results
bool result = qi::phrase_parse(begin, end, p, boost::spirit::ascii::space, m); // returns true if successful
if(begin != end) {
std::cout << "Parsing failed!" << std::endl;
} else {
std::cout << "Parsing succeeded!" << std::endl;
}
std::cout << m["AB"] << "\t" << m["E"] << "\t" << m["G"] << "\t" << m["J K L"] << std::endl;
return 0;
}
这样的结果或多或少是我需要的:
Parsing succeeded!
CD F HI M N O
我要解决的最后一个问题是 A [B C] -> F [D E]
这样的案例.
有什么方法可以将它们作为四个独立的键值对 ("A", "F"), ("A", "D E"), ("B C", "F"), ("B C", "D E")
进入我的std::map<std::string, std::string> m
?
或者将其解析为 std::map<std::vector<std::string>, std::vector<std::string> >
可能更容易每个std::vector<std::string>
保存所有键/值?
例如:
in: "{A [B C] -> F [D E], C ->E,B->Z}"
out: { ({"A", "B C"}, {"F", "D E"}), ({"C"}, {"E"}), ({"B"}, {"Z"}) }
感谢您的帮助!
最佳答案
我认为您非常接近您的目标,所以我将跳过组合部分 :-)解析器将做它应该做的事情……检查语法并标记数据,然后它将键、值和输出映射 ( multimap ) 参数传递到 phoenix 函数 inserter
中,您可以在其中插入无论您在 map 中需要什么
( multimap )
#if __cplusplus >= 201103L
#define BOOST_RESULT_OF_USE_DECLTYPE
#endif
#define BOOST_SPIRIT_USE_PHOENIX_V3
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <iostream>
#include <algorithm>
#include <iterator>
#include <iomanip>
#include <vector>
#include <map>
namespace qi = boost::spirit::qi;
namespace ascii=boost::spirit::ascii;
typedef std::map< std::string,std::string > TMap;
//typedef std::multimap< std::string,std::string > TMap;
struct SMapInsert
{
template <typename Arg1,typename Arg2,typename Arg3>
struct result
{
typedef void type;
};
template <typename Arg1,typename Arg2,typename Arg3>
void operator()( Arg1&out, Arg2&keys, Arg3&vals ) const
{
std::cout << "Keys:" << std::endl;
for( const auto &key : keys )
std::cout << std::left << "`" << key << "`" << std::endl;
std::cout << "Vals:" << std::endl;
for( const auto &val : vals )
std::cout << std::left << "`" << val << "`" << std::endl;
// your map here...
// out.insert
}
};
boost::phoenix::function< SMapInsert > inserter;
int main()
{
std::string input = "{A [B C] -> F [D E], C ->E,B->Z}";
TMap data;
std::string::const_iterator iter = input.begin();
std::string::const_iterator last = input.end();
qi::rule< std::string::const_iterator,std::string() > token=+qi::alnum;
qi::rule< std::string::const_iterator,ascii::space_type,std::vector< std::string >() >
keyOrvalue = +( token | ( '[' >> qi::lexeme[ +qi::char_("a-zA-Z0-9 ") ] >> ']' ) );
qi::rule< std::string::const_iterator,ascii::space_type, TMap() >
root = '{' >> ( ( keyOrvalue >> "->" >> keyOrvalue )[ inserter( qi::_val, qi::_1, qi::_2 ) ] ) % ',' >> '}';
std::cout << "input: `" << input << "`" << std::endl;
if( qi::phrase_parse( iter, last, root, ascii::space, data ) && iter==last )
{
for( const auto &keyValue : data )
std::cout << std::left << std::setw(10) << keyValue.first << std::setw(10) << keyValue.second << std::endl;
}
else
std::cout << "parsing failed:" << std::string( iter,last ) << std::endl;
return 0;
}
关于c++ - 使用 boost::spirit::qi::phrase_parse 将值存储在 std::map<std::string, std::string> 中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18579591/
我正在尝试从一个 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 map 的 m
我是 Haskell 的新手,我认为函数 map map和 map.map在 Haskell 中是一样的。 我的终端给了我两种不同的类型, (map.map) :: (a -> b) -> [[a]
我的目标是创建一个 map 的 map ,这样我就可以通过它的键检索外部 map 的信息,然后通过它们的键访问它的“内部” map 。 但是,当我得到每个内部映射时,我最初创建的映射变成了一个对象,我
如何使用 Java8 编写以下代码? for (Entry> entry : data.entrySet()) { Map value = entry.getValue(); if (valu
我有覆盖整个南非的图片。它们为Tiff格式,并已将坐标嵌入其中。我正在尝试拍摄这些图像(大约20张图像),并将它们用作我的iPhone应用程序中的地图叠加层。我的问题在于(准确地)将地图切成图块。 我
所以我有 2 std::map s >一个是“旧的”,一个是“新的”,我想知道哪些文件被删除了,这样就能够遍历差异并对 shared_ptr 做一些事情。这样的事情可能吗?如何做到? 最佳答案 虽然
是否可以将当前查看的 google.maps.Map 转换为静态图像链接,以便我可以获取图像并将其嵌入到 PDF 中? 我在 map 上添加了一些带有自定义图标的标记,所以我不确定这是否真的可行。 如
你能帮我吗 Java Streams ? 从标题可以看出我需要合并List>>进入Map> . 列表表示为List>>看起来像: [ { "USER_1":{
对于 idAndTags 的第二个条目,内部映射被打乱,但第一个条目则不然 第一次接近! for (Map.Entry> entryOne : idAndTags.entrySet()) {
我将从我的代码开始,因为它应该更容易理解我想要做什么: @function get-color($color, $lightness) { @return map-get(map-get($col
我过去曾在许多网站上使用过 Google map ,但遇到了以前从未遇到过的问题。 map 窗口正在显示,但它只显示左上角的 map 片段,以及之后的任何内容(即使我在周围导航时),右侧也不会加载任何
众所周知,这些 map ,无论是常规街道 map 还是卫星 map ,在中国的特定地区都无法正确排列。那么哪个 map 排列正确,是卫星 map 还是默认街道 map ?一些网站表明卫星 map 是正
在拖尾事件之后,我面临着获取此处 map 中的 map 边界的问题。我需要新的经纬度来在新更改的视口(viewport)中获取一些项目/点。我只是想在拖动结束时获得谷歌地图map.getBounds(
我想做的是通过 ajax API 显示以英国邮政编码为中心的小型 bing 生成 map 。我相信这是可能的;我在 Bing map 文档中找不到如何将英国邮政编码转换为可以插入 map Ajax 控
我有一个 List我想转换成的 e Map>其中外部字符串应为“Name”,内部字符串应为“Domain”。 Name Id Domain e(0) - Emp1, 1, Insuran
我的第 2 部分:https://stackoverflow.com/questions/21780627/c-map-of-maps-typedef-doubts-queries 然后我继续创建 I
是否可以在 1 行中使用 Java8 编写以下所有 null 和空字符串检查? Map> data = new HashMap<>(holdings.rowMap()); Set>> entrySet
我正在审查一个项目的旧代码,并使用 Map 的 Map 的 Map 获得了如下数据结构(3 层 map ): // data structure Map>>> tagTree
这可能是一种不好的做法,但我还没有找到更好的解决方案来解决我的问题。所以我有这张 map // Map>> private Map>> properties; 我想初始化它,这样我就不会得到 Null
我们在 JDK 1.7 中使用 HashMap,我在使用 SonarQube 进行代码审查时遇到了一些问题。 请考虑以下示例: public class SerializationTest imple
我是一名优秀的程序员,十分优秀!