Используя Boost Spirit Qi qi :: iter_pos, но не могу получить значение - PullRequest
0 голосов
/ 02 мая 2019

Мне нужно отслеживать положение некоторых элементов в тексте, который я анализирую с помощью Boost Spirit Qi.Я нашел этот пример и адаптирован так:

#include <iostream>
#include <string>
#include <boost/spirit/home/qi.hpp>
#include <boost/spirit/repository/include/qi_iter_pos.hpp>
#include <boost/spirit/include/phoenix.hpp>

namespace phx = boost::phoenix;
namespace qi = boost::spirit::qi;

template<typename Iterator>
struct CurrentPos 
{
    CurrentPos() 
    {
        save_start_pos = qi::omit[boost::spirit::repository::qi::iter_pos[
                phx::bind(&CurrentPos::setStartPos, this, qi::_1)]];

        current_pos = boost::spirit::repository::qi::iter_pos[
                qi::_val = phx::bind(&CurrentPos::getCurrentPos, this, qi::_1)];
    }

    qi::rule<Iterator> save_start_pos;
    qi::rule<Iterator, std::size_t()> current_pos;

private:
    void setStartPos(const Iterator &iterator) 
    {
        start_pos_ = iterator;
    }

    std::size_t getCurrentPos(const Iterator &iterator) 
    {
        return std::distance(start_pos_, iterator);
    }

    Iterator start_pos_;
};

using InfoTuple = std::tuple<std::size_t, std::uint32_t>;

struct Header
{
    std::uint32_t x;
    InfoTuple y;
};

BOOST_FUSION_ADAPT_STRUCT(
    Header,
    (std::uint32_t, x)
    (InfoTuple, y)
)

template<typename Iterator>
struct HeaderParse
    : boost::spirit::qi::grammar<Iterator, Header()>
{
    HeaderParse()
        : HeaderParse::base_type(_start)
    {
        using boost::spirit::qi::uint_parser;

        _thing = (current_pos.current_pos >> uint_parser<std::uint32_t, 10, 1, 3>());
        _start = current_pos.save_start_pos 
            >> '<' 
            >> uint_parser<std::uint32_t, 10, 1, 3>() 
            >> '>'
            >> _thing;
    }   

    qi::rule<Iterator, InfoTuple()> _thing; 
    qi::rule<Iterator, Header()> _start;

    CurrentPos<Iterator> current_pos;
};

int main()
{
    const std::string d1 = "<13>937";

    const HeaderParse<std::string::const_iterator> parser;
    Header header;

    std::string::const_iterator begin = d1.begin();
    std::string::const_iterator end = d1.end();

    assert(boost::spirit::qi::parse(begin, end, parser, header));
    assert(begin == end);

    std::cout << "x    : " << header.x << std::endl;
    std::cout << "y    : " << std::get<1>(header.y) << std::endl;
    std::cout << "y pos: " << std::get<0>(header.y) << std::endl;
}

Когда я его запускаю, я вижу следующий вывод:

$> ./testme
x    : 13
y    : 0
y pos: 4

По какой-то причине он будетне захватывает значение 937.

Я попробовал это на Coliru, и сначала он не компилировался, и теперь я продолжаю получать "срок действия истек".Смотрите здесь: https://coliru.stacked -crooked.com / a / 724c7fcd296c8803 Но это заставляет меня задуматься, действительно ли это так.Я использую clang, а Coliru использует gcc.

Может кто-нибудь подсказать, почему это не сработает?

1 Ответ

1 голос
/ 03 мая 2019

Ваш пример не компилируется для меня из-за отсутствия интеграционного заголовка Fusion std::tuple (boost/fusion/include/std_tuple.hpp). После добавления этого примера компилируется и выводится ожидаемый результат. https://wandbox.org/permlink/wOf8JxnKKeBGCihk

P.S: Я бы посоветовал не беспокоиться о вычислении смещения в вашем парсере, он ничего не спасет, хранение самого итератора имеет тот же объем памяти, но меньше сложности и боли.

...