boost :: spirit解析带有分隔符号的double

won*_*ime 2 c++ boost-spirit-qi

我正在使用boost :: spirit来将文本解析为双精度形式,其演唱形式可能与数字之间用空格分隔。

使用或滥用real_policies时,我找到了解决方案,但不确定是否有更简单的方法来实现。有人可以给我提示吗?

现场例子

这是相关的代码片段:

template <typename T>
struct real_with_separated_sign_policies : boost::spirit::qi::real_policies<T>
{
    // allow skipping chars between a possible sign and a folling real number
    template <typename Iterator>
    static bool parse_sign(Iterator& first, Iterator const& last)
    {
        bool ret = qi::extract_sign(first, last);
        if (ret)
            qi::parse(first, last, *qi::lit(' '));
        return ret;
    }
};

template <typename Iterator, typename Skipper>
struct RealWithSeparatedSignParser
    : qi::grammar<Iterator, double(), Skipper>
{
    boost::spirit::qi::real_parser<double, real_with_separated_sign_policies<double> > RealWithSeparatedSignValue;

    RealWithSeparatedSignParser() : RealWithSeparatedSignParser::base_type(start)
    {
        start %= RealWithSeparatedSignValue;
    }

    qi::rule<Iterator, double(), Skipper> start;
};

int main() {
    std::string str = " -  1.234 ";
Run Code Online (Sandbox Code Playgroud)

seh*_*ehe 5

我会像您在这里一样进行此操作。您应该考虑还原first迭代器,以防进一步解析失败。

您可能需要仔细检查multi_pass<>适应的迭代器上的刷新语义(我认为这是可以的,因为real_parser无论如何,包装必须能够在失败时回退)。


当然,考虑到语法的简单性,可以减少样本,但这并不是我想的重点。

这是一个较短的演示,显示了更通用的跳过策略(blank_type默认情况下使用):

Live On Coliru

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

namespace qi = boost::spirit::qi;

template <typename T, typename Skipper = qi::blank_type>
struct skip_after_sign_policies : boost::spirit::qi::real_policies<T> {
    // allow skipping chars between a possible sign and a folling real number
    template <typename Iterator>
    static bool parse_sign(Iterator& first, Iterator const& last) {
        return qi::extract_sign(first, last)
            && qi::phrase_parse(first, last, qi::eps, Skipper {});
    }
};

int main() {
    qi::real_parser<double, skip_after_sign_policies<double> > grammar;

    std::string const str = " -  1.234 ";

    auto it = str.begin();
    double value;
    bool ok = phrase_parse(it, str.end(), grammar, qi::space, value);

    std::cout << std::boolalpha << ok << " " << value;
    if (it != str.end())
        std::cout << "Remaining: '" << std::string(it, str.end()) << "'\n";
}
Run Code Online (Sandbox Code Playgroud)

版画

true -1.234
Run Code Online (Sandbox Code Playgroud)