- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我一直在用 boost::spirit 实现一个解析器,它需要在输出时生成 google::protobuf 生成的类。
我试图跟随 page作为背景。不幸的是,我不能使用属性语法,因为 google::protobuf 生成的类只提供 set/get 方法。因此,我尝试使用 boost::phoenix 进行延迟绑定(bind),但我不确定如何从 A 类绑定(bind) add_param() 方法(参见下面的代码和第 99 行的注释):
#define BOOST_SPIRIT_DEBUG
#include <string>
#include <iostream>
#include <sstream>
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <boost/fusion/include/std_pair.hpp>
namespace qi = boost::spirit::qi;
namespace phx = boost::phoenix;
class A
{
public:
A(): _i(0) {}
A(int i) : _i(i)
{
std::cout << "i=" << i << std::endl;
}
void set_i(int i)
{
_i = i;
}
void add_param(const std::string& value)
{
_params.push_back(value);
}
std::string to_string() const
{
std::stringstream ss;
ss << "_i=[" << _i << "], _params=[";
for (auto& p : _params)
ss << p << ",";
ss << "]" << std::endl;
return ss.str();
}
private:
int _i;
std::vector < std::string > _params;
};
class B
{
public:
B() : _id(), _av() {}
B(int id, const std::vector<A>& av) : _id(id), _av(av) {}
void set_id(int id)
{
_id = id;
}
void add_A(const A& a)
{
_av.push_back(a);
}
std::string to_string() const
{
std::stringstream ss;
ss << "_id=[" << _id << "], _av=[";
for (auto& av : _av)
ss << av.to_string() << ",";
ss << "]" << std::endl;
return ss.str();
}
private:
int _id;
std::vector<A> _av;
};
namespace Utils
{
int mul2(int i)
{
return i * 2;
}
}
template <typename Iterator, typename Skipper>
struct grammar_B: qi::grammar<Iterator, Skipper>
{
grammar_B(B& ctx) : grammar_B::base_type(start)
{
// A
i_rule = qi::int_[ qi::_val = phx::bind(Utils::mul2, qi::_1) ];
param_rule = *(qi::char_ - ',' - '!');
a_rule = (qi::lit("$")
>> i_rule
>> qi::lit(":")
>> *(param_rule >> +(qi::lit(","))) // how to bind to A::add_param() ?
>> qi::lit("!")) [ qi::_val = phx::construct<A>(qi::_1) ];
// B
id_rule = qi::int_[ phx::bind(&B::set_id, phx::ref(ctx), qi::_1) ];
start %= id_rule >> qi::lit("=") >> a_rule;
BOOST_SPIRIT_DEBUG_NODE(id_rule);
BOOST_SPIRIT_DEBUG_NODE(i_rule);
BOOST_SPIRIT_DEBUG_NODE(param_rule);
}
private:
qi::rule<Iterator, Skipper> start;
qi::rule<Iterator, int, Skipper> i_rule;
qi::rule<Iterator, std::string(), Skipper> param_rule;
qi::rule<Iterator, A(), Skipper> a_rule;
qi::rule<Iterator, int, Skipper> id_rule;
};
int main(int argc, char* argv[])
{
std::string input = "1=$100:param1,param2!";
B ctx;
grammar_B<decltype(std::begin(input)), qi::space_type> p(ctx);
auto f(std::begin(input)), l(std::end(input));
if (qi::phrase_parse(f, l, p, qi::space))
std::cout << "Success: " << ctx.to_string() << std::endl;
else
std::cout << "failed" << std::endl;
return 0;
}
现在,我有 2 个问题:
1) 我的方法是否可以通过一般的延迟绑定(bind)来实现?
2) 如何将 std::vector<> 生成的属性绑定(bind)到 A::add_param() 之类的方法?
--
谢谢
最佳答案
好的,我终于能够实现我需要的东西了: Coliry Example
关于c++ - Spirit::Qi :延迟绑定(bind)到 google-protobuf,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39973024/
假设我们要解析一个内积表达式并得到结果。 "SUM({1, 2, 3} .* {4, 5, 6})" qi::_1 和 qi::_2 非常方便地在解析器中引用第 i 个属性。 void Test(st
无法弄清楚为什么这条规则 unary_msg 不起作用,它说属性类型是 qi::unused_type 但这对我来说毫无意义。为什么 boost 会这样折磨我? template struct g3:
从测试中我可以得到 qi::uint_parser()与 qi::uint_ 相同.他们从 0 解析整数至 UINT_MAX . 我不明白的是qi::uint_parser需要 std::numeri
bool EqnExprEvaluator::SetEqn(const std::string& eqnStr) { typedef std::string::const_iterator
我一直在使用 Boost 库中的 Spirit Qi 用 C++ 编写语法。作为该语言的新手,很难习惯该库的语法和怪癖,但现在我有点了解它的工作原理。 我在使用 qi::lexeme 时遇到错误。我的
我需要跟踪一些项目在我使用提升灵气解析的文本中的位置。我找到了 this example并适应这样: #include #include #include #include #include
我想做的是在运行时从 ABNF 语法文件创建一个解析器。我已经在 qi::grammar 中实现了所有 ABNF 规则,如下所示: typedef /*qi::rule or struct conta
我在boost_1.59中使用spirit来解析类c语言(名为stone,最初由java编写)。但是我在使用 boost::spirit::qi 解析 c 风格的字符串时遇到了麻烦。整个代码位于 co
我正在用 Boost.Spirit 做一个 IRC 消息解析器,但是当我尝试解析一个输入时我遇到了一个(很长的)错误。我遵循了“Roman Numerals”示例。此外,我将 g++4.7 与 -st
如何解析可能包含 double 或 int 的字符串,具体取决于是否设置了点。例如。 6.0是double类型,6是int类型。规则是 rule,skipper> r = qi::double_|qi
在 boost::spirit::traits::transform_attribute 中指示解析失败的正确方法是什么?我可以抛出任何旧的异常,还是它要我做的特定事情? namespace boos
我正在尝试使用 qi 创建通用解析器元素,因为不幸的是(必须支持 MSVC)无法使用 X3。这个想法是有一个模板化的结构: template struct parse_type; 我可以这样使用: t
我正在尝试创建一个返回 function 的规则通过柯里化(Currying) Phoenix 表达式构造。例如, start = int_[_val = xxx]; rule start; 应该做什
我正在解析一个文本文件,可能有几 GB 的大小,由以下几行组成: 11 0.1 14 0.78 532 -3.5 基本上,每行一个整数和一个 float 。整数应该是有序的并且是非负的。我想验证数据是
我想提出一个让我陷入困境的主题,并提出了一个关于齐::符号。 这一切都始于我查看新的野兽图书馆并阅读 atutorial example 它以一个从 http 路径猜测 mime 类型的函数开始扩展。
我有一段 boost::spirit::qi 代码可以匹配 "M7.x.y.z" 或 "M7.x.y.z ",但我想要一个在前一个输入上失败的解析器。 我想我需要在其中插入 qi::lexeme[]
我正在尝试为一种语言的多个子集定义一些 Boost::spirit::qi 解析器,代码重复最少。为此,我创建了一些基本的规则构建函数。原来的解析器工作正常,但一旦我开始使用组合函数,我的解析器似乎就
假设我有一个简单的类型foo: struct foo { void bar(int) { // do something } }; 我想使用 Boost.Qi 来解析字符串中的整数字段,并使用结果值调
(免责声明,我正在学习提升精神) 我正在尝试解析这样的表达式:F( 1 ) 并希望将 1 作为字符串(“1”而不是数字(与 qi::int_ 一起使用))。 我试过这样的事情(这是错误的,但也许我的方
假设我有两个用逗号分隔的 double 来解析返回它们的总和。在 Haskell 中我可能会这样做: import Data.Attoparsec.Text import Data.Text (pac
我是一名优秀的程序员,十分优秀!