【问题标题】:boost::spirit char/string mixingboost::spirit 字符/字符串混合
【发布时间】:2011-08-16 03:28:15
【问题描述】:

我正在尝试使用 boost::spirit 来解析字符标记,并且遇到了很大的困难。这是我正在处理的示例代码:

#include <boost/spirit/include/qi.hpp>
#include <iostream>
#include <string>

using namespace std;
namespace qi = boost::spirit::qi;
namespace ascii = boost::spirit::ascii;

template <typename Iterator>
struct tok_parser : qi::grammar<Iterator, string(), ascii::space_type>
{
  tok_parser() : tok_parser::base_type(start)
  {
    tok1   = qi::char_("AB");
    tok2   = qi::char_("12");
    mytoks = (qi::lit("A1") | qi::lit("A2") | qi::lit("B1") | qi::lit("B2"));
    start  = mytoks;
    //start  = tok1 >> tok2;  // error 1
    //start  = +mytoks;       // error 2
  }
  qi::rule<Iterator, string(), ascii::space_type> start;
  qi::rule<Iterator, string(), ascii::space_type> mytoks;
  qi::rule<Iterator, char, ascii::space_type> tok1;
  qi::rule<Iterator, char, ascii::space_type> tok2;
};

int main(int argc, char** argv)
{
  tok_parser<string::const_iterator> g; // Our grammar
  string str = argv[1];
  string::const_iterator iter = str.begin();
  string::const_iterator end = str.end();
  bool r = phrase_parse(iter, end, g, boost::spirit::ascii::space, str);

  if (r && iter == end)
      cout << "Parsing succeeded\n";
  else
      cout << "Parsing failed\n";

  return 0;
}

错误 1:

我想做的是创建两个令牌规则,让我可以解析所有 A1、A2、B1、B2,而不必列出所有文字排列。正如代码一样,它编译并识别目标标记。但是,如果我尝试根据 tok1 和 tok2 规则构建解析器,它不会编译生成有关类型不匹配的错误:

error: invalid static_cast from type âboost::spirit::qi::char_class<boost::spirit::tag::char_code<boost::spirit::tag::space, boost::spirit::char_encoding::ascii> >â to type âcharâ

我认为这与尝试用字符创建字符串有关。我玩过任意数量的类型猜测排列,但都失败了。

错误 2:

另外,我想允许解析任意数量的标记,不幸的是,前面加上 + 运算符也不能按预期工作。

有什么建议吗?

【问题讨论】:

    标签: c++ parsing boost-spirit


    【解决方案1】:

    对于规则,属性总是需要使用函数声明语法来指定:

    qi::rule<Iterator, char(), ascii::space_type> tok1;  
    qi::rule<Iterator, char(), ascii::space_type> tok2;
    

    这应该可以解决您的编译器问题。

    Wrt 你的排列问题:我建议做这样的事情:

    rule<Iterator, string(), ascii::space_type> r = char_("AB") >> char_("12");
    

    将匹配“A1”、“A2”、“B1”或“B2”。

    创建另一个规则允许所需的重复:

    rule<Iterator, vector<string>(), ascii::space_type> rs = +r;
    

    【讨论】:

    • 感谢您的提示,精神是相当简单/复杂/精彩/可怕的图书馆。
    猜你喜欢
    • 1970-01-01
    • 2023-03-30
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-05-04
    • 2015-11-24
    相关资源
    最近更新 更多