2013-10-02 47 views
4

我希望能够解析一个数字,以存储它的原始来源,并跟踪其来源中的位置,保存它在结构本身。助推精神与源解析

这是我到目前为止有:

#include <boost/config/warning_disable.hpp> 
#include <boost/spirit/include/qi.hpp> 
#include <boost/spirit/include/phoenix_core.hpp> 
#include <boost/spirit/include/phoenix_operator.hpp> 
#include <boost/spirit/include/phoenix_object.hpp> 
#include <boost/spirit/home/support/iterators/line_pos_iterator.hpp> 
#include <boost/fusion/include/adapt_struct.hpp> 
#include <boost/fusion/include/io.hpp> 

#include <iostream> 
#include <iomanip> 
#include <ios> 
#include <string> 
#include <complex> 

#include <boost/spirit/include/phoenix_fusion.hpp> 
#include <boost/spirit/include/phoenix_stl.hpp> 

struct Position 
{ 
    Position() 
     : line(-1) 
    { 
    } 

    size_t line; 
}; 

struct Number : public Position 
{ 
    Number() 
     : Position() 
     , value(-1) 
     , source() 
    { 
    } 

    unsigned value; 
    std::string source; 
}; 

using namespace boost::spirit; 

BOOST_FUSION_ADAPT_STRUCT(Number, 
          (unsigned, value) 
          (std::string, source) 
          (size_t,  line) 
         ); 

template <typename Iterator> 
struct source_hex : qi::grammar<Iterator, Number()> 
{ 
    source_hex() : source_hex::base_type(start) 
    { 
     using qi::eps; 
     using qi::hex; 
     using qi::lit; 
     using qi::raw; 
     using qi::_val; 
     using qi::_1; 
     using ascii::char_; 

     namespace phx = boost::phoenix; 
     using phx::at_c; 
     using phx::begin; 
     using phx::end; 
     using phx::construct; 

     start = raw[ (lit("0x") | lit("0X")) 
        >> hex [at_c<0>(_val) = _1] 
        ][at_c<2>(_val) = get_line(begin(_1))] 
        [at_c<1>(_val) = construct<std::string>(begin(_1), end(_1))] 

     ; 
    } 

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

和测试代码:

typedef line_pos_iterator<std::string::const_iterator> Iterator; 
source_hex<Iterator> g; 
Iterator iter(str.begin()); 
Iterator end(str.end()); 

Number number; 
bool r = parse(iter, end, g, number); 
if (r && iter == end) { 
    std::cout << number.line << ": 0x" << std::setw(8) << std::setfill('0') << std::hex << number.value << " // " << number.source << "\n"; 
} else 
    std::cout << "Parsing failed\n"; 

什么,我没有得到就是为什么行的迭代器:

[at_c<2>(_val) = get_line(begin(_1))] 

不是line_pos_iterator,即使这是我正在使用的解析器。 我会欣赏解释以及如何解决问题的想法 - 无论如何。

+1

,显然我在做什么是完全关闭 - 因为get_line是语法 – gsf

+1

你需要调用'get_line'为“懒惰”的建设过程中调用functor(凤凰男演员)。见[这个答案](http://stackoverflow.com/questions/8358975/cross-platform-way-to-get-line-number-of-an-ini-file-where-given-option-was-foun/ 8365427#8365427)的例子(Inifile解析器),它使用它 – sehe

回答

6

看一看

#include <boost/spirit/repository/include/qi_iter_pos.hpp> 

根据定义,这种直接暴露位置作为属性的分析器。让我在几分钟内添加一个例子。

编辑我发现很难鞋拔iter_pos到您的样品没有“假设”的东西,改变你的数据类型的布局。我非常青睐这一点(我一直在努力失去语义操作)。但时间有限。

这里有一个小帮手,你可以用它来解决您的问题:

struct get_line_f 
{ 
    template <typename> struct result { typedef size_t type; }; 
    template <typename It> size_t operator()(It const& pos_iter) const 
    { 
     return get_line(pos_iter); 
    } 
}; 

^多态的演员,因为这样使用:

start = raw[ qi::no_case["0x"] >> hex [at_c<0>(_val) = _1] ] 
       [ 
        at_c<1>(_val) = construct<std::string>(begin(_1), end(_1)), 
        at_c<2>(_val) = get_line_(begin(_1)) 
       ] 
    ; 

    // with 

boost::phoenix::function<get_line_f> get_line_; 

注意我改了几个小问题。

完全运行演示与输出:Live On Coliru