Обработка исключений в семантических действиях

Рассмотрим следующий парсер:

class test
{
public:
  static test from_string(const string &str); //throws!
};

template <typename Iterator = string::const_iterator>
struct test_parser : grammar<Iterator, test(), blank_type>
{
  test_parser() : test_parser::base_type(query)
  {
    query = id[_val = phx::bind(&test::from_string, qi::_1)];
    id = lexeme[*char_("a-zA-Z_0-9")];
  }
  rule<Iterator, test(), blank_type> query;
  rule<Iterator, string(), blank_type> id;
};

Я хотел бы поймать исключения, которые test::from_string может бросить и провалить матч по исключению. Я не мог найти прямой способ сделать это, поэтому я пытаюсь использовать функцию "адаптера", которая бы явно принимала контекст. Но как получить доступ к контексту и как прикрепить такое действие к грамматике? Пожалуйста, смотрите вопросы в коде:

template<class Context>
void match_test(const string &attr, Context &context, bool &mFlag)
{
  try
  {
    test t = test::from_string(attr);
    // how do I access the context to put t into _val?
  }
  catch(...)
  {
    mFlag = false;
  }
}

//...
test_parser() : test_parser::base_type(query)
{
  query = id[?match_test<?>? /*how to instantiate and use the above semantic action?*/];
  id = lexeme[*char_("a-zA-Z_0-9")];
}

1 ответ

Решение

Как сказал комментатор, используйте

    query = id[
            phx::try_ [
                qi::_val = phx::bind(&test::from_string, qi::_1)
            ].catch_all [ 
                qi::_pass = false 
            ]
        ];

Посмотреть это в прямом эфире на Coliru

Версия, которая компилируется даже с BOOST_SPIRIT_USE_PHOENIX_V3: Жить на Coliru

    query = id[
            phx::try_ [
                qi::_val = phx::bind(&test::from_string, qi::_1)
            ].catch_all [ 
                qi::_pass = false 
            ],
            qi::_pass = qi::_pass // to appease the spirit expression compilation gods
        ];
Другие вопросы по тегам