Мне было интересно, есть ли способ в Boost.Spirit.Qi динамически комбинировать произвольное количество правил во время выполнения.Внутренняя работа Boost.Spirit все еще остается для меня загадкой, но поскольку правила реализованы в виде объектов, это представляется возможным.Моя мотивация - сделать некоторые части моей грамматики легко расширяемыми.
Рассмотрим следующий надуманный пример:
namespace qi = boost::spirit::qi;
namespace px = boost::phoenix;
typedef std::string::const_iterator iterator_t;
template<typename Expr>
inline bool parse_full(const std::string& input, const Expr& expr)
{
iterator_t first(input.begin()), last(input.end());
bool result = qi::phrase_parse(first, last, expr, boost::spirit::ascii::space);
return first == input.end() && result;
}
void no_op() {}
int main(int argc, char *argv[])
{
int attr = -1;
// "Static" version - Works fine!
/*
qi::rule<iterator_t, void(int&)> grammar;
qi::rule<iterator_t, void(int&)> ruleA = qi::char_('a')[qi::_r1 = px::val(0)];
qi::rule<iterator_t, void(int&)> ruleB = qi::char_('b')[qi::_r1 = px::val(1)];
qi::rule<iterator_t, void(int&)> ruleC = qi::char_('c')[qi::_r1 = px::val(2)];
grammar =
ruleA(qi::_r1) | //[no_op]
ruleB(qi::_r1) | //[no_op]
ruleC(qi::_r1); //[no_op]
*/
// "Dynamic" version - Does not compile! :(
std::vector<qi::rule<iterator_t, void(int&)>> rules;
rules.push_back(qi::char_('a')[qi::_r1 = px::val(0)]);
rules.push_back(qi::char_('b')[qi::_r1 = px::val(1)]);
rules.push_back(qi::char_('c')[qi::_r1 = px::val(2)]);
std::vector<qi::rule<iterator_t, void(int&)>>::iterator i(rules.begin()), last(rules.end());
qi::rule<iterator_t, void(int&)> grammar;
grammar = (*i)(qi::_r1);
for(++i; i!=last; ++i)
{
grammar = grammar.copy() | (*i)(qi::_r1);
}
// Tests
if(parse_full("a", grammar(px::ref(attr)))) std::cout << attr << std::endl;
if(parse_full("b", grammar(px::ref(attr)))) std::cout << attr << std::endl;
if(parse_full("c", grammar(px::ref(attr)))) std::cout << attr << std::endl;
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
return 0;
}
Ошибка Visual Studio 2010:
error C2440: 'initializing' : cannot convert from 'boost::fusion::void_' to 'int &'
Я подозреваю, что это вызвано тем, что унаследованный атрибут не передан grammar.copy()
.К сожалению, я не мог найти простой способ сделать это, поэтому я выбрал обходной путь.В результате у меня есть одна последняя версия (и я уже хотел бы поблагодарить всех, кто остался до сих пор!).Это на самом деле кажется работающим:
// "Dynamic" version - Kind of works! :-/
std::vector<qi::rule<iterator_t, void(int&)>> rules;
rules.push_back(qi::char_('a')[qi::_r1 = px::val(0)]);
rules.push_back(qi::char_('b')[qi::_r1 = px::val(1)]);
rules.push_back(qi::char_('c')[qi::_r1 = px::val(2)]);
std::vector<qi::rule<iterator_t, void(int&)>>::iterator i(rules.begin()), last(rules.end());
qi::rule<iterator_t, int()> temp;
temp = (*i)(qi::_val); //[no_op]
for(++i; i!=last; ++i)
{
temp = temp.copy() | (*i)(qi::_val); //[no_op]
}
qi::rule<iterator_t, void(int&)> grammar;
grammar = temp[qi::_r1 = qi::_1];
Однако, когда я присоединяю простое семантическое действие (такое как «[no_op]»), поведение становится действительно странным.Вместо того, чтобы печатать 0,1,2, как раньше, он печатает 0,0,2.Итак, мне интересно, что я пытаюсь достичь в результате неопределенного поведения?Это ошибка?Или вполне возможно, я просто что-то использую (например, семантические действия?) Неправильно?